内存管理机制
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)
- 标记-整理,标记过程一样,然后将使用了的内存整理到一起,然后清除边界以外的内存
- 分代收集算法.分新生代和老年代两种,分别采用不同的策略来进行垃圾回收
- 新生代采用复制算法.因为新生代对象回收率比较高
- 老年代采用标记清除或者标记整理算法来回收.