《深入理解Java虚拟机》阅读笔记

内存管理机制

Java内存区域与内存溢出异常

运行时数据区域

Java虚拟机在执行程序的过程中,会将内存划分为方法区,堆,虚拟机栈,本地方法栈,程序计数器

  • 程序计数器:当前线程执行字节码的行号指示器。唯一一个没有规定OOM的区域,线程私有。
  • Java虚拟机栈:生命周期与线程相同,线程私有。它描述的是Java方法执行的内存模型。Java方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。其中,局部变量表所需要的内存空间在编译期间就会完成分配。
  • 本地方法栈:和虚拟机栈相似,主要为native方法服务。线程私有。
  • :堆是被所有线程共享的一块内存区域。Java虚拟机规范中描述到:所有的对象实例以及数组都要在堆上分配。其中还规定:Java堆可以处于物理上不连续的内存空间中,大小可自主选择固定或可扩展
  • 方法区:别名Non-Heap(非堆)。同样也是各个线程共享的一块内存区域。用于存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。同样可以处于物理上不连续的内存空间中,大小可自主选择固定或可扩展。Java虚拟机规范规定:当方法区无法满足内存分配需求的时候,将抛出OOM
    • 运行时常量池:隶属于方法区。Class文件中除了有类的版本、字段、方法、接口等描述信息,还有常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放。运行时常量池相对于Class文件常量池的另外一个重要特征是具备动态性。如String类的intern()方法。同样,当常量池内存不足时会抛出OOM。
  • 直接内存:它并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。在JDK1.4中新加入了NIO类,引入了一种基于Channel和Buffer的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样可显著提高性能。

GC算法

  • 引用计数(废弃,因为无法解决循环引用的问题)

  • 目前采用的都是可达性算法.通过GCRoots来遍历引用链.可做为GCRoots的有:

    • 虚拟机栈中的引用对象
    • 方法区中类静态属性引用的对象
    • 方法区中常量引用的对象
    • JNI引用的对象
  • 算法类型

    • 标记清除算法
    • 复制算法(分为Eden和2个Survivor,比例为8:1:1)
    • 标记-整理,标记过程一样,然后将使用了的内存整理到一起,然后清除边界以外的内存
    • 分代收集算法.分新生代和老年代两种,分别采用不同的策略来进行垃圾回收
      • 新生代采用复制算法.因为新生代对象回收率比较高
      • 老年代采用标记清除或者标记整理算法来回收.

你可能感兴趣的:(Java,java)