JVM内存空间划分与作用详解

2020-12-18  本文已影响0人  得力小泡泡

JVM运行时内存数据区域:


image.png
注意:

一个进程都有一套方法区和堆
一个线程都有一套Java虚拟机栈和程序计数器

一、虚拟机栈:

表示Java方法执行的内存模型,每调用一个方法就会为每个方法生成一个栈帧(Stack Frame),用来存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法被调用和完成的过程,都对应一个栈帧从虚拟机栈上入栈和出栈的过程。虚拟机栈的生命周期和线程是相同的。不同的线程使用不同的栈

不过我们这里要说的栈还不是虚拟机栈,而是虚拟机栈里,一个栈帧的操作数栈。因为,我这里只是演示一个方法而已,这一个方法其实就是一个栈帧

image.png

一个栈帧主要由四部分组成:

二、程序计数器

2.1 简述

程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器。

2.2 作用

大家都知道,Java程序从源文件创建到程序运行要经过两大步骤:

  1. 源文件由编译器编译成字节码(ByteCode)。
  2. 字节码由java虚拟机解释运行。

字节码解释器工作时就是通过改变这个 程序计数器 的值来选取下一条需要执行的字节码指令

当一个线程执行到一半的时候核心切到了第二个线程了,势必要记录当前线程要挂起到什么位置上,此信息就是记录在程序计数器上的,而每一个程序计数器里面的线程执行的信息其实是线程私有的,也就是说A线程是不能得到B线程的程序计数器的。如果一个程序正在执行某一个Java方法,程序计数器记录的就是正在执行字节码对应的地址而已。

三、本地方法栈

2.1 简述

本地方法栈(Native Method Stack)与虚拟机栈所发挥的作用是非常相似的,其 区别 不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的Native方法服务。

四、Java堆

4.1 描述

对于大多数应用来说,Java堆(Java Heap)是Java虚拟机所管理的内存中最大的一块。Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。

4.2 作用

1、此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。
2、Java堆是垃圾收集器管理的主要区域,因此很多时候也被称做“GC堆”(Garbage Collected Heap)。

4.2 Java堆的组成:

① 如果从内存回收的角度看,由于现在收集器基本都是采用的分代收集算法,所以Java堆中还可以细分为:新生代和老年代;
② 如果从内存分配的角度看,线程共享的Java堆中可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB)。

五、方法区

5.1 描述

同 Java 堆一样,方法区也是全局共享的一块内存区域。

5.2 作用

1、它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。 类信息和常量池,其中常量池有定义的方法信息、变量信息、常量(字面量和符号引用) **
2、与Java堆不一样之处在于:
方法区并不是垃圾回收的主要区域。**
相对而言,垃圾回收在这个区域是比较少出现的,这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载。 所以会将方法区称为永久代,但是从JDK1.8开始,已经彻底废弃了永久代了,使用元空间(Meta Space)代替。
3、当前的商业JVM都有实现方法区的GC,主要是回收两部分内容:废弃常量与无用类。
4、类加收需要满足如下3个条件【条件是极其苛刻的】:

5、在大量使用反射、动态代理、CGLib等字节码框架、动态生成JSP以及OSGi这类频繁自定义ClassLoader的场景都需要JVM具备类卸载的支持以保证方法区不会溢出。

5.3 运行时常量池

运行时常量池(Runtime Constant Pool)也是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池(Constant Pool Table),用于存放编译期生成的各种字面量和符号引用,这部分将在类加载后进入方法区运行时常量池中存放。

六、直接内存

Direct Memory,它并不是由JVM所管理的一块区域,而是由系统所管辖的,只不过是JVM向系统申请了这块内存。

上一篇下一篇

猜你喜欢

热点阅读