JVM内存区域:
1、虚拟机栈:
方法执行时的内存模型,是线程私有的,生命周期跟线程相同。方法执行时入栈,方法执行完出栈,出栈相当于清空数据,所以这块不需要GC。
2、本地方法栈:
与虚拟机栈功能非常类似,区别在于虚拟机栈为虚拟机执行java方法时服务,而本地方法栈为虚拟机执行本地方法时服务,也不需要GC.
3、程序计数器:
线程独有,可以看作是当前线程执行的字节码的指示器。
4、堆:
对象实例和数组都在堆上分配,GC主要对这两类数据进行回收。
垃圾回收主要方法:
- 标记清除算法
先标记出相应的可回收对象,对可回收的对象进行回收,不用移动数据,有内存碎片。
- 复制算法
把堆分成两块区域A和B,区域A负责分配对象,把存活的对象标记出来复制到B区,最后把A区对象全部清理,这样就解决了内存碎片的问题了。问题:比如堆分配了500M内存,只有250M可用,移动对象效率低下。
- 标记整理法
在标记清除法的基础上添加了一个整理的过程,将所有的存活对象往一端移动,紧邻排列,清理另一端的区域,这样就解决了内存碎片的问题。但是每进行一次垃圾清除都要频繁移动存活对象,效率低下。
分代收集算法:
大部分的对象都在很短的时间内被回收了(Minor GC就会被回收),分代收集算法根据对象存活周期的不同将堆分成新生代和老年代,默认比例1:2,新生代分为Eden区,S0,S1,三者的比例为8:1:1,新生代的GC称为Young GC(也叫Minor GC),老年代发生的GC称为Old GC (也叫Full GC)。
1、当Eden区将满时,触发Minor GC,存活的对象会被移到S0区;当下一次Minor GC时,会把Eden区和S0(S1)中存活的对象移到S1,同时对象年龄+1,当对象的年龄达到了设定的阈值,则晋升到老年代。
2、如果老年代满了,会触发Full GC,Full GC会同时回收新生代和老年代(即对整个堆进行GC),造成挺大的性能开销。
GC期间,只有垃圾回收器线程在工作,其他工作线程则被挂起。(为啥GC时工作线程会被挂起?一边收垃圾,另一边丢垃圾,收拾不干净)
新生代的垃圾回收器:Serial, ParNew, ParallelScavenge
老年代的垃圾回收器:CMS, Serial Old, Parallel Old
同时在新老生代的垃圾回收器:G1
如果是运行在桌面环境处于Client模式的,则用Serial + Serial Old 收集器;
如果需要响应时间快,则用ParNew +CMS的搭配;
G1回收器也需要根据吞吐量等要求适当调整相应的JVM参数。
没有最牛的技术,只有最合适的使用场景。