JVM 内存区域详解

一.JVM内存分为 五个部分:

1)方法区;2java虚拟机栈;3)本地方法栈;4)堆;5)程序计数器

 

1)程序计数器:可以看做是当前线程所执行的字节码的行号指示器。下一条需要执行的字节码指令,分之,循环,跳转,异常处理,线程恢复等基础功能都需要依赖这个计数器来完成。

每个线程都有一个独立的程序计数器,各条线程之间计数器互不影响,独立存储,为“线程私有的内存”。


2)Java虚拟机栈:也是线程私有的,生命周期和线程相同。每个方法在执行的时候都会创建一个栈帧,用于存储局部变量表,操作数栈,动态链接,方法出口等信息。每一个方法从调用到执行完成的过程,就对应的这一个栈帧在虚拟机栈中从入栈到出栈的过程。

我们平时所说的Java栈内存一般指Java虚拟机栈中的局部变量表部分。

局部变量表存放了各种基本数据类型(booleanint等),对象引用类型(不是对象,而是其引用,存放相关地址)

局部表量表的内存大小分配在程序编译时就确定了,在方法运行期间不会改变其大小。


3)本地方法栈:与虚拟机栈所发挥的作用是非常相似的。虚拟机栈为执行java方法服务,而本地方法栈为虚拟机使用到的Native方法服务。

一个Native Method就是一个java调用非java代码的接口。一个Native Method是这样一个java的方法:该方法的实现由非java语言实现,比如C。这个特征并非java所特有,很多其它的编程语言都有这一机制,比如在C++中,你可以用extern "C"告知C++编译器去调用一个C的函数。)


4) Java堆(Heap):是被所有线程共享的一块线程区域,在虚拟机启动时创建。

    堆内存区域的唯一目的就是存放对象实例(包括所有的对象实例,数组等)

Java堆是垃圾收集器管理的主要区域,又称“GC堆”。

Java堆可细分为新生代和老年代;再细分可分为Eden空间,From Survivor空间,To Survivor空间等。

线程共享的Java堆中可能划分出多个线程私有的分配缓冲区。


5) 方法区(Non-Heap):是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据。

这区域的内存回收目标主要是针对常量池的回收和对类型的卸载。

运行时常量池:是方法区的一部分。用于存放编译期生成的各种字面量和符号引用。这部分内容将在类加载后进入方法区的运行时常量池中存放。


二.Java对象的内存布局

分为3块区域:1)对象头2)实例数据3)对齐填充


1)对象头:一部分是用于存储对象自身的运行时数据,如哈希码,GC分带年龄,锁状态标识,线程持有的锁,偏向线程ID,偏向时间戳等。
另外一部分是类型指针,即对象指向他的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。如果对象是一个Java数组,那在对象头中还必须有一块用于记录数组长度的数据。
2)实例数据部分:对象真正存储的有效信息,也是在程序代码中所定义的各种类型的字段内容。无论是从父类继承下来的,还是在子类中定义的,都需要记录起来。父类定义的变量会出现在子类之前。
3)对齐填充:不是必须存在的,因为Java对象大小必须是8字节的整数倍,不够的需要对齐填充来补全。


三.JVM 内存溢出

分为1Java堆溢出2)虚拟机栈和本地方法栈溢出3)方法区和运行时常量池溢出

1)Java堆溢出:Java对用于存储对象实例,只要不断的创建对象,并且保证GC Roots到对象之间有可达路径来避免垃圾回收机制清除这些对象,那么在对象数量达到最大堆容量限制后就会产生内存溢出异常。(测试方法:不断的new对象并addList,此时创建的对象不满足垃圾回收机制)

虚拟机栈和本地方法栈溢出:栈深度超过虚拟机允许最大深度;扩展栈无法申请足够空间。

3)方法区和运行时常量池溢出:方法区用于存放Class的相关信息,如类名,访问修饰符,常量池,字段描述,方法描述等。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值