JAVA 内存分为以下几个部分
- 1.程序计数器
作用
:看做是当前线程所执行的字节码的行号指示器,字节码解释器工作时,就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。我们的分支,循环,跳转,异常处理,线程恢复等基础功能都需要依赖这个计数器完成。
作用域
:线程私有,一个处理器或者一个内核在任何时候只会执行一条(多线程会来回切换),为了线程切换后能恢复到正确的执行位置,所以每个线程都有独立的程序计数器
拾遗
:计数器记录的是正在执行的虚拟机字节指令的地址,但是如果是native方法,则计数器的值为空,此区域是虚拟机规范中唯一没有规定任何的oom异常的区域。生命周期跟线程一样
- 2.java虚拟机栈(我们常说的栈)
作用
:一个线程执行一个方法就往栈里面存放一个栈帧,方法结束代表出栈,栈帧存放局部变量表,操作栈,动态的链接,方法出口等信息
作用域
:线程私有,生命周期跟线程一样
拾遗
:局部变量表存放编译期可知道的各种基本数据类型,对象引用(不同的虚拟机或者采用直接指向对象起始地址的引用指针,或者指向一个代表对象的句柄)和returnAddress指向 一条字节码指令的地址。long和double是64位长度占据局部变量表的2个slot,其余只是一个slot。局部变量表所需要的内存空间在编译期间已经完成分配,在方法运行期间不会改变局部变量表的大小。stackoverflowerror和outofmemoryerror 都会抛出
- 3.本地方法栈
作用
:其与虚拟机栈作用类似,只是前者作用于java方法后者作用于native方法。
作用域
:线程私有,生命周期跟线程一样
拾遗
:stackoverflowerror和outofmemoryerror 都会抛出
- 4.java 堆(我们常说的堆)
作用
:存放对象的实例是内存中最大的一块,被所有线程共享(每个线程在Java堆中预先分配一小块内存,称为本地线程分配缓冲(Thread Local Allocation Buffer , TLAB)这样分配对象不需要加锁)
作用域
:所有线程共享,生命周期跟JVM一致
拾遗
:也有栈上分配,标量替换,这些不一定会把对象分配在堆上,其实内存中占用最大的一块,整个堆类似于map,可以在物理上不连续。
- 5.方法区(一般也包含在我们说的堆里面)
作用
:存储已被虚拟机架子的类信息,常量,静态变量,即时编译器编译后的代码等数据。
作用域
:线程共享,生命周期与jvm一致
拾遗
:虽然其属于堆的逻辑部分,但是我们称呼其为non-heap,也有人把方法区成为永久代,本质上两者并不等价,只是因为hotspot使用永久代来实现方法区,对于其他虚拟机是不存在永久代,后期已经把方法区办到本地内存。这个区域的内存可以不连续,也可以不需要实现垃圾回收。如果要垃圾回收 也是针对常量池的回收和对象类的卸载。如果想卸载一个类:1.该类所有的实例都已经被GC。2.加载该类的ClassLoader已经被GC。3.该类的Java.lang.Class 对象没有在任何地方被引用。
- 6.运行时常量池
作用
:class文件中除了有类的版本,字段,方法,接口等描述信息外,还有一项是常量池,用于存放编译器生成的各种字母量和符号引用,这部分内容将在类加载后存放到方法区的常量池中。也会存放直接引用
作用域
:线程共享,生命周期与jvm一致
拾遗
:属于方法区的一部分
- 7.直接内存
作用
:避免了在java堆和native堆中来回的复制数据
作用域
:线程共享
拾遗
:比如netty都使用到了堆外内存
网友评论