虚拟机自动内存管理机制
内存泄漏、内存溢出
Java 编译器(编译)
Java 虚拟机:执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。
运行时数据区:方法区、堆、虚拟机栈、本地方法栈、程序计数器
方法区和堆是所有线程共享的数据区
虚拟机栈、本地方法栈、程序计数器是线程隔离的数据区(线程私有内存)
程序计数器:一块比较小的内存空间,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。
虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。
局部变量表所需的内存空间在编译期间完成分配,在方法运行期间不会改变局部变量表的大小。
HotSpot虚拟机直接把本地方法栈、和虚拟机栈合二为一。
所有的对象实例以及数组都要在堆上分配。(随着JIT编译器发展与逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化发生,所有对象都分配在堆上也渐渐变得不是那么绝对了)
Java堆是垃圾收集器管理的主要区域,因此很多时候也被称作“GC堆”,现在收集器基本都采用分代收集算法,所有Java堆还可以细分为:新生代和老年代。
线程共享的Java堆可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer, TLAB),通过配置-XX:+/-UseTLAB参数设定是否使用
为每一个线程在伊甸园空间分配一块独享的空间,这样每个线程只访问他们自己的TLAB空间,再与bump-the-pointer技术结合可以在不加锁的情况下分配内存。
-Xmx -Xms
方法区(别名:Non-Heap,非堆;也叫“永久代”,本质上两者不等价)
用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
-XX:MaxPermSize
目前已经发布的JDK1.7的HotSpot中,已经把原本放在永久代的字符串常量池移出。
CAS配上失败重试的方式保证更新操作的原子性。
对象内存布局:对象头、实例数据和对齐填充。
强引用、软引用、弱引用、虚引用
generator
Full gc yong gc
Java堆分为新生代和老年代。根据对象存活周期分类,新生代对象存活率低,老年代对象存活率高。
垃圾收集算法
标记-清除算法、复制算法、标记-整理算法、分代收集算法
老年代空间的GC事件基本上是在空间已满时发生,执行的过程根据GC类型不同而不同