首页 > 代码库 > 一步一步学JVM-运行时数据区域
一步一步学JVM-运行时数据区域
程序计数器(Program Counter Register)
像我们平时读书一样,当我们在去做别的事情之前,我们会对我们读到什么地方了做一个标记,方便我们再回来的时候接着重新读。如果这本书有很多人读呢?那么每个人都会对自己读到了哪里做一个标记。(标记都是个人自己保存)
那么程序计数器也是一样的,同样的代码,可能会被多个线程执行,那么每个线程都要记住自己执行到哪行代码了。这样在线程再次切换执行的时候,知道代码接着从哪里执行。
所以程序计数器是线程私有的,每个线程都会程序计数器。对于正在执行的Native方法,这个计数器值为空。
栈
虚拟机栈(VM Stack)
与程序计数器一样,Java虚拟机栈也是线程私有的。每个线程执行的时候都会创建一个栈帧(Stack Frame)。用来存放局部变量表、操作数栈、动态链接、方法出口等信息。每个方法的调用到执行完成的过程,都对应着一个栈帧在虚拟机中入栈到出栈的过程。
局部变量表
局部变量表中存放了编译器可知的各种数据类型,和对象引用类型。其中除了64位长度的long和double类型会占用2个局部变量空间(Slot),其余的数据类型只占用1个。所以每个方法需要分配多大的局部变量表空间是完全确定的,在编译期间就完成分配。在方法运行期间不会发生改变。
本地方法栈(Native Method Stack)
与虚拟机栈一样,只是虚拟机栈是为虚拟机执行的Java方法服务,而本地方法栈为虚拟机中使用到的Native方法服务。
堆
Java堆,是被所有线程共享的区域,在虚拟机启动时创建。用来存放对象示例。
随着对象的创建,Java堆中的对象实例会越来越多,会造成内存溢出。在这些对象中,有百分之八十都是使用完之后就不用了,所以我们可以对这些对象进行回收。就引出了垃圾回收机制,对堆中没用的对象进行清理。为了方便对象清理,又把堆分为了新生代和老年代。其中新生代又可以详细分为Eden区、From Survive区和To Survive区。
方法区
在方法区存储了类的信息,静态变量、常量等数据。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它有一个别名Non-Heap(非堆),目的是为了和Java堆区分开。
在HotSpot虚拟机上,方法区也被称为“永久代”,这与Java虚拟机的垃圾回收有关。但是这样设计并不好,更加容易导致内存溢出问题。HotSpot虚拟机现在也有放弃永久代并逐步改为采用Native Memory来实现方法区的规划。从JDK1.7的HotSpot中,已经把原本在永久代的字符串常量池移出。
运行时常量池
运行时常量池也是方法区的一部分。用来存放编译器生成的各种字面量和符号引用。Java语言并不要求常量池一定只有编译器才能产生,运行期间也可能将新的常量放入池中。利用的最多的就是String的intern()方法。
直接内存
直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分内存被频繁使用,也可能导致内存溢出异常出现。
一步一步学JVM-运行时数据区域