这里默认指的是 HotSpot 虚拟机
运行时数据区域JVM运行过程中管理的内存可划分成若干个不同的数据区域。JDK 1.8 和之前的版本略有不同。
线程私有的:程序计数器,虚拟机栈,本地方法栈
线程共享的:堆,方法区,直接内存 (非运行时数据区的一部分)
程序计数器主要有两个作用:
1.流程控制,如:顺序执行、选择、循环、异常处理。
2.在多线程的情况下记录当前线程执行的位置,当线程被切换回来的时候能够知道该线程上次运行到哪儿了。
虚拟机栈每个线程都有各自的虚拟机栈。
虚拟机栈是由一个个栈帧组成,每个栈帧中都拥有:局部变量表(主要存放编译期可知的基本数据类型和对象引用。)、操作数栈、动态链接、方法出口信息。每次方法调用都会有一个对应的栈帧被压入栈中,每次方法结束后,都会有一个栈帧被弹出。
本地方法栈与虚拟机栈作用类似,不同的是本地方法栈是为虚拟机使用到的 Native 方法服务
堆堆在虚拟机启动时创建,该区域是用来存放对象实例的,几乎所有的对象实例以及数组都在这里分配内存。
从JDK1.7开始,如果方法中的对象引用没有被返回或者未被外面使用,那么对象可以直接在栈上分配内存。
在JDK1.7及之前版本,堆内存被通常分为下面三部分:
新生代(又可分为Eden区,两个Survivor区)
老年代
永久代(1.8之后被移除)
用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等。HotSpot虚拟机在JDK1.8之前使用了永久代这种方式实现方法区,JDK8及以后采用元空间这种直接内存来取代
为什么要将永久代替换为元空间呢?
1.JVM在永久代设置了固定大小的上限,而元空间只受本机可用内存的限制
2.元空间里面存放的是类的元数据,这样要加载多少类的元数据就可以由系统的实际可用空间来控制,可以加载更多的类
3.在 JDK1.8合并 HotSpot 和 JRockit 的代码时, JRockit 没有永久代,合并之后也没有必要额外的设置一个永久代
运行时常量池运行时常量池是方法区的一部分。Class 文件中除了有类的版本、属性、方法、接口等描述信息外,还有常量池表(用于存放编译期生成的各种字面量和符号引用)
JDK1.7 之前运行时常量池包含字符串常量池存放在方法区,JDK1.7 字符串常量池被单独从方法区拿到了堆中,运行时常量池剩下的东西还在方法区,JDK1.8字符串常量池还在堆, 运行时常量池还在方法区, 只不过方法区的实现从永久代变成了元空间
直接内存直接内存并不是虚拟机运行时数据区的一部分,内存分配受到本机总内存大小以及处理器寻址空间的限制。



