在JVM
中程序寄存器、Java虚拟机栈、本地办法栈,这三个区是随着线程的创立而创立,随着线程完结而销毁。
其实就是这三个的生命周期和线程的生命周期一样。都是每个线程公有的。
每次办法的调用就会向栈里入栈一个栈帧,办法调用完结,跟着就出栈。
对象也是有生命周期的,所以对于不须要的对象要进行必要的分明,否则长此以往,咱们的内存就被一点一点的耗费完。
明天来学习,如何判断对象是否曾经能够被回收?以及回收有哪些算法?
如何判断对象已死?
援用计数法
给对象增加一个援用计数器,每当一个中央援用它object时技术加1,援用失去当前就减1,计数为0阐明不再援用。
- 长处:实现简略,断定效率高;
- 毛病:无奈解决对象互相循环援用的问题,对象A中援用了对象B,对象B中援用对象A。
public class A { public B b; }public class B { public C c; }public class C { public A a; }public class Test{ private void test(){ A a = new A(); B b = new B(); C c = new C(); a.b=b; b.c=c; c.a=a; }}
可达性剖析算法
当一个对象到GC Roots
没有援用链相连,即就是GC Roots
到这个对象不可达时,证实对象不可用。
GC Roots
品种:
Java 线程中,以后所有正在被调用的办法的援用类型参数、局部变量、长期值等。也就是与咱们栈帧相干的各种援用。
所有以后被加载的 Java 类。
Java 类的援用类型动态变量。
运行时常量池里的援用类型常量(String 或 Class 类型)。
JVM 外部数据结构的一些援用,比方 sun.jvm.hotspot.memory.Universe 类。
用于同步的监控对象,比方调用了对象的 wait() 办法。
public class Test{ private void test(C c){ A a = new A(); B b = new B(); a.b=b; //这里的a/b/c都是GC Root; }}
对象的援用类型
强援用:
User user=new User();
咱们开发中应用最多的对象援用形式。
特点:咱们平时典型编码Object obj = new Object()中的obj就是强援用。
通过关键字new创立的对象所关联的援用就是强援用。
当JVM内存空间有余,JVM宁愿抛出OutOfMemoryError运行时谬误(OOM),使程序异样终止,也不会靠随便回收具备强援用的“存活”对象来解决内存不足的问题。
对于一个一般的对象,如果没有其余的援用关系,只有超过了援用的作用域或者显式地将相应(强)援用赋值为 null,就是能够被垃圾收集的了,具体回收机会还是要看垃圾收集策略。
软援用:
SoftReference object=new SoftReference(new Object());
特点:软援用通过SoftReference类实现。软援用的生命周期比强援用短一些。
只有当 JVM 认为内存不足时,才会去试图回收软援用指向的对象:即JVM 会确保在抛出 OutOfMemoryError 之前,清理软援用指向的对象。
软援用能够和一个援用队列(ReferenceQueue)联结应用,如果软援用所援用的对象被垃圾回收器回收,Java虚拟机就会把这个软援用退出到与之关联的援用队列中。
后续,咱们能够调用ReferenceQueue的poll()办法来查看是否有它所关怀的对象被回收。如果队列为空,将返回一个null,否则该办法返回队列中后面的一个Reference对象。
利用场景:软援用通常用来实现内存敏感的缓存。如果还有闲暇内存,就能够临时保留缓存,当内存不足时清理掉,这样就保障了应用缓存的同时,不会耗尽内存。
弱援用:
WeakReference object=new WeakReference (new Object();
ThreadLocal中有应用。
弱援用通过WeakReference类实现。弱援用的生命周期比软援用短。
在垃圾回收器线程扫描它所管辖的内存区域的过程中,一旦发现了具备弱援用的对象,不论以后内存空间足够与否,都会回收它的内存。因为垃圾回收器是一个优先级很低的线程,因而不肯定会很快回收弱援用的对象。
弱援用能够和一个援用队列(ReferenceQueue)联结应用,如果弱援用所援用的对象被垃圾回收,Java虚拟机就会把这个弱援用退出到与之关联的援用队列中。利用场景:弱利用同样可用于内存敏感的缓存。
虚援用:
简直没见过应用, ReferenceQueue 、PhantomReference。
finalize办法
这个办法就有点相似于“某个人被判了死刑,然而不肯定会死”的情景。
即便在可达性剖析算法中不可达的对象,也并非肯定是“非死不可”的,这时候他们临时处于“缓刑”阶段,真正宣告一个对象死亡至多要经验两个阶段:
1、如果对象在可达性剖析算法中不可达,那么它会被第一次标记并进行一次刷选,刷选的条件是是否须要执行finalize()办法(当对象没有笼罩finalize()或者finalize()办法曾经执行过了(对象的此办法只会执行一次)),虚拟机将这两种状况都会视为没有必要执行)。
2、如果这个对象有必要执行finalize()办法会将其放入F-Queue队列中,稍后GC将对F-Queue队列进行第二次标记,如果在重写finalize()办法中将对象本人赋值给某个类变量或者对象的成员变量,那么第二次标记时候就会将它移出“行将回收”的汇合。
办法区的回收
在堆中,尤其是在新生代中,一次垃圾回收个别能够回收 70% ~ 95% 的空间,而办法区的垃圾收集效率远低于此。
办法区垃圾回收次要两局部内容:废除的常量和无用的类。
垃圾回收算法
标记-革除
第一步:就是找出沉闷的对象。咱们反复强调 GC 过程是逆向的, 依据 GC Roots 遍历所有的可达对象,这个过程,就叫作标记。
第二步:除了下面标记进去的对象以外,其余的都革除掉。
- 毛病:标记和革除效率不高,标记和革除之后会产生大量不间断的内存碎片
复制
新生代应用,新生代分中Eden:S0:S1
= 8:1:1,其中前面的1:1就是用来复制的。
当其中一块内存应用完了,就将还存活的对象复制到另外一块下面,而后把曾经应用过的内存空间一次革除掉。
个别对象调配都是进入新生代的eden区,如果Minor GC
还存活则进入S0
区,S0
和S1
一直对象进行复制。对象存活年龄最大默认是15,大对象进来可能因为新生代不存在间断空间,所以会间接接入老年代。任何应用都有新生代的10%是空着的。
- 毛病:对象存活率高时,复制效率会较低,节约内存。
标记整顿
它的次要思路,就是挪动所有存活的对象,且依照内存地址程序顺次排列,而后将末端内存地址当前的内存全副回收。 然而须要留神,这只是一个现实状态。对象的援用关系个别都是非常复杂的,咱们这里不对具体的算法进行形容。咱们只须要理解,从效率上来说,个别整顿算法是要低于复制算法的。这个算法是躲避了内存碎片和内存节约。
让所有存活的对象都向一端挪动,而后间接清理掉端边界以外的内存。
从下面的三个算法来看,其实没有相对最好的回收算法,只有最适宜的算法。
STW
STW
=Stop The world,字面翻译过去就是整个世界都进行了。
在JVM中也有这么个说法,就是STW,是指JVM垃圾收集器在收集垃圾对象的时候,其余所有线程都被挂起(除了垃圾收集器之外),JVM中一种全局暂停景象。
----全局进展,想想就很可怕,所有的Java代码进行执行,native代码能够执行,然而不能与JVM进行交互,这些基本上都是因为GC引起的。
然而也还有另外的几种场景也能够导致STW:
1.Garbage collection pauses2.Code deoptimization3.Flushing code cacheClass redefinition (e.g. hot swap or instrumentation)4.Biased lock revocation5.Various debug operation (e.g. deadlock check or stacktrace dump)