JVM--02

Java虚拟机内存管理:

共享:

  方法区:存储运行时常量池、已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据

  java堆:存储对象实例

线程独占区:

  虚拟机栈:存放方法运行时所需的数据、成为栈帧

  本地方法栈:为jvm所调用到的Nativate即本地方法服务

  程序计数器:记录当前线程所执行到的字节码的行号

程序计数器:

较小的内存区域 当前线程所执行的字节码行号指示器  

如果是java方法 记录当前字节码指令的地址   如果是native方法 值为undefined  没有oom异常的区域

goto java中唯一的一个保留字

虚拟机栈:

  java方法执行的动态内存模型

  栈帧:

    每个方法执行,都会创建一个栈帧,伴随方法创建到执行完成。存储局部变量表、操作数栈、动态链接、方法出口等。

  局部变量表:

    存放编译器可知的各种基本数据类型,引用类型,returnAddress类型  64位的long 和double占用两个slot(局部变量空间)其余只占用一个

    局部变量表的内存空间在编译期完成分配,当进入一个方法时,这个方法需要在帧分配多少内存是固定的,运行期间不会改变局部变量表的大小。

  大小:

    如果线程请求的栈深度大于虚拟机所允许的深度,将发生Stack Overflow  例如递归方法没有出口

    如果扩展时无法申请到足够的空间将发生oom

本地方法栈:

  本地方法栈为虚拟机使用到的native方法服务

  也会出现 Stack Overflow和oom

Java堆:

  存放对象实例

  jvm管理的最大的内存区域

  垃圾收集器管理的主要区域

  新生代、老年代、Eden空间

  (逃逸分析、栈上分配、标量替换)

  OOM异常

方法区:(非堆 Non-heap)

  线程共享

  存储虚拟机加载的类信息、常量、静态变量,即时编译器编译后的代码等数据

  类版本  字段  方法  接口

  方法区和永久代

  垃圾回收在方法区的行为:针对常量池的回收和类型的卸载

  异常的定义:OOM

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

  存放编译期生成的各种字面量和符号引用,将在类加载后进入方法区的运行时常量池中存放

  也会把翻译出来的直接引用存放到运行时常量池中

  运行期间也可以将新的常量放入常量池中 String类的intern方法

  OOM异常

intern方法是native方法

直接内存:(Direct Memory)

  不是jvm的一部分  

  OOM异常

  NIO (New Input/Output):引入一种基于通道与缓冲区的I/O方式 使用native函数库直接分配堆外内存,通过一个存储在java heap中的DirectByteBuffer对象作为这块内存的引用进行操作

  由DirectMemory导致的内存溢出,在heap dump文件中不会看到明显的异常

java对象的结构:

  Header(对象头)

    自身运行时数据(Mark Word):hash值  GC分代年龄  锁状态标志  线程持有的锁  偏向线程id  偏向时间戳

    类型指针:数组(还有一个记录数组长度的数据)  指向对象元数据的指针

  InstanceData:

    long、double分在一起  short。char分到一起  相同长度分一起

  Padding

    占位符  

对象创建:分配空间

  指针碰撞:Serial、ParNew等使用compact过程的收集器

  空闲列表:CMS 基于Mark-Sweep算法

对象访问定位:

  通过栈上的reference数据操作堆上的具体对象

  使用句柄:java堆中划分出句柄池 保存实例对象的地址  java栈中的引用地址不需要改变 只改变句柄池

  使用指针 hotspot采用的方式:直接指针  速度快

到对象实例的指针和到对象类型数据的指针

 

    

猜你喜欢

转载自www.cnblogs.com/zhy-study/p/9291922.html