JVM内存模型

程序计数器(线程私有):

也有称作PC寄存器。每个线程都有一个私有的程序计数器,程序计数器存放的是当前线程所执行的字节码的行号指示器。当CPU需要执行指令时,需要从程序计数器中得到当前需要执行的指令所在存储单元的地址,然后根据得到的地址获取到指令,在得到指令之后,程序计数器便自动加1或者根据转移指针得到下一条指令的地址,如此循环,直至执行完所有的指令。程序计数器也控制着程序的分支、循环、跳转、异常、恢复等。Java多线程是通过线程轮流切换并分配处理器执行时间来实现的,一个核心在一个时刻只能执行一条线程的指令,所以为了线程切换后能恢复到原来位置,每个线程都需要一个程序计数器(线程私有)

 

Java虚拟机栈/Java栈(线程私有):

1、在创建线程的时候会创建对应线程的虚拟机栈。栈中存放的是一个个的栈帧,当有一个方法被执行时,就会创建一个栈帧,栈帧存储着局部变量表、操作栈、动态链接、方法返回地址等。

2、局部变量表包含基本类型、对象引用(引用指针)、returnAddress类型等;
且局部变量表所需的内存空间在编译期间完成分配,在方法运行期间不会改变局部变量表的大小。

3、如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常,但大部分虚拟机的栈深度可动态扩展,当扩展无法申请到足够的内存时OutOfMemoryError

 

本地方法栈(线程私有):

1、与Java栈的作用和原理相似,区别是:Java栈为虚拟机执行Java方法服务,而本地方法栈为虚拟机用到的Native方法服务.

2、有的虚拟机(如Sun HotSpot虚拟机)直接将本地方法栈和虚拟机栈合二为一

3、本地方法栈一样会抛出StackOverflowError和OutOfMemoryError异常

 

Java堆(线程共享):

1、Java堆是Java虚拟机所管理的内存中最大的一块

2、Java堆在虚拟机启动时创建,目的是存放对象实例,几乎所有的对象实例和数组都在这里分配

3、Java堆是垃圾收集器管理的主要区域,因此很多时候也被成为“GC堆”

4、从内存回收角度来看,Java堆可以细分为新生代和老年代(分代收集算法);再细致一点有Eden控件、From Survicor空间、To Survicor空间。

5、从内存分配角度来看,线程共享的Java堆可能划分出多个线程私有的分配缓冲区。

6、Java堆可以处于物理上不连续的内存空间,可扩展;如果堆中没有内存完成实例分配,且无法再扩展时,将会抛出OutOfMemoryError异常

 

方法区(线程共享):

1、用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等

2、不需要连续的内存、可扩展,也可以选择不实现垃圾收集,垃圾收集在这个区域较少出现

3、这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载,但是回收的条件苛刻,可能因为对该区域未完全回收而导致内存泄漏

4、当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常

 

运行时常量池(方法区的一部分):

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

2、一般来说,除了保存Class文件中描述的符号引用外,还会把翻译出来的直接引用

也存储在运行时常量池中。

3、运行期间也可能将新的常量放入池中,如String类的intern()方法

4、常量池无法再申请到内存的时候,将抛出OutOfMemoryError异常

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值