的引用_java中的强引用,软引用,弱引用,虚引用
一對(duì)象回收算法
Java在GC時(shí)判斷對(duì)象是否存活有兩種方式;第一種是引用計(jì)數(shù)方式,第二種是可達(dá)性分析算法;
引用計(jì)數(shù)器算法:
在對(duì)象中添加一個(gè)引用計(jì)數(shù)器,每當(dāng)有一個(gè)地方引用它時(shí),計(jì)數(shù)器值就加一;當(dāng)引用失效時(shí),計(jì)數(shù)器值就減一;任何時(shí)刻計(jì)數(shù)器為零的對(duì)象就是不可用對(duì)象
額外占用空間,高效;可達(dá)性分析算法:
從gc roots 往下搜索,走過(guò)的路徑稱為引用鏈,如果引用鏈斷開(kāi)說(shuō)明對(duì)象不可達(dá)標(biāo)記為可回收對(duì)象;如果引用鏈連續(xù)說(shuō)明此對(duì)象可達(dá)標(biāo)記為不可回收對(duì)象。如圖所示,object4,object5,object7都是被標(biāo)記為可回收對(duì)象。
jdk1.2版本之前 如果一個(gè)對(duì)象沒(méi)有被引用,則無(wú)法使用這個(gè)對(duì)象;jdk1.2版本之后對(duì)象被分為4個(gè)引用級(jí)別,由高到低依次為:強(qiáng)引用、軟引用、弱引用和虛引用;開(kāi)發(fā)人員可以利用對(duì)象的引用級(jí)別靈活控制對(duì)象的生命周期。
二強(qiáng)引用(StrongReference)
強(qiáng)引用是指引用賦值,即Objecto bj = new Object()。無(wú)論任何情況下,只要強(qiáng)引用關(guān)系還存在,垃圾收集器就永遠(yuǎn)不會(huì)回收掉被引用的對(duì)象。
如下方式都是強(qiáng)引用
Object obj = new Object(); String str = "zszxz";如果一個(gè)對(duì)象具有強(qiáng)引用,不會(huì)被垃圾回收器回收。當(dāng)內(nèi)存空間不足,Java虛擬機(jī)寧愿拋出OutOfMemoryError錯(cuò)誤,使程序異常終止,也不回收對(duì)象,故通常如果對(duì)象不用的時(shí)候就需要手動(dòng)賦值為 null, 方便垃圾回收器回收
示例
String str = null;當(dāng)然 通常對(duì)象不會(huì)手動(dòng)賦值為null, 只有比較大的對(duì)象集合時(shí)就需要手動(dòng)賦值,比如 ArrayList 里面的 clear() 方法。
public void clear() {modCount++; ?// clear to let GC do its workfor (int i = 0; i < size; i++)elementData[i] = null; ?size = 0;}注:養(yǎng)成手動(dòng)釋放內(nèi)存能提示程序性能三軟引用(SoftReference)
軟引用是用來(lái)描述一些還有用,但非必須的對(duì)象。只被軟引用關(guān)聯(lián)著的對(duì)象,在系統(tǒng)將要發(fā)生內(nèi)存溢出異常前,會(huì)把這些對(duì)象列進(jìn)回收范圍之中進(jìn)行第二次回收,如果這次回收還沒(méi)有足夠的內(nèi)存,才會(huì)拋出內(nèi)存溢出異常;
源碼:
// 時(shí)間戳,右垃圾回收器更新static private long clock; ?// 時(shí)間戳,通過(guò)get方法調(diào)用更新,選擇軟引用時(shí)將被清除,VM可以使用這個(gè)字段,并非必須private long timestamp; ? ?public SoftReference(T referent) {// 構(gòu)造器,給對(duì)象注入一個(gè)軟引用super(referent);this.timestamp = clock;} ?// 給對(duì)象注入一個(gè)軟引用,并且注入隊(duì)列public SoftReference(T referent, ReferenceQueue<? super T> q) {super(referent, q);this.timestamp = clock;} ?// 返回對(duì)象的引用,如果引用對(duì)象已經(jīng)被清除或者垃圾回收,則返回Nullpublic T get() {T o = super.get();if (o != null && this.timestamp != clock)this.timestamp = clock;return o;}如下方式,即使發(fā)生gc,只要內(nèi)存沒(méi)有溢出就不會(huì)回收軟引用
public static void main(String[] args) throws InterruptedException {// 給對(duì)象注入軟引用SoftReference<String> reference = new SoftReference<>(new String("zszxz"));// zszxzSystem.out.println(reference.get());// 調(diào)用GCSystem.gc();TimeUnit.SECONDS.sleep(5);// zszxzSystem.out.println(reference.get());}適用于緩存場(chǎng)景四弱引用(WeakReference)
當(dāng)垃圾收集器開(kāi)始工作,無(wú)論當(dāng)前內(nèi)存是否足夠,都會(huì)回收掉被弱引用關(guān)聯(lián)的對(duì)象;
源碼:
public class WeakReference<T> extends Reference<T> {// 給對(duì)象注入弱引用public WeakReference(T referent) {super(referent);} ?// 給對(duì)象注入一個(gè)弱引用并注入引用隊(duì)列public WeakReference(T referent, ReferenceQueue<? super T> q) {super(referent, q);} ? }如下方式,一旦GC調(diào)用則會(huì)回收對(duì)象。
public static void main(String[] args) throws InterruptedException {WeakReference<String> weakReference = new WeakReference<>(new String("zszxz"));// zszxzSystem.out.println(weakReference.get());// 調(diào)用GCSystem.gc();TimeUnit.SECONDS.sleep(5);// zszxzSystem.out.println(weakReference.get());}五虛引用(PhantomReference)
為一個(gè)對(duì)象設(shè)置虛引用關(guān)聯(lián)的唯一目的只是為了能在這個(gè)對(duì)象被收集器回收時(shí)收到一個(gè)系統(tǒng)通知,除此之外沒(méi)有任何作用,不決定對(duì)象的生命周期;
源碼
public class PhantomReference<T> extends Reference<T> { ?// 獲得引用public T get() {return null;} ?// 創(chuàng)建虛引用public PhantomReference(T referent, ReferenceQueue<? super T> q) {super(referent, q);} ? }如下示例,即使不調(diào)用GC,引用也是為null。
public static void main(String[] args) throws InterruptedException {// 引用隊(duì)列ReferenceQueue queue = new ReferenceQueue();PhantomReference reference = new PhantomReference(new String("zszxz"),queue);// nullSystem.out.println(reference.get());}六總結(jié)
- 強(qiáng)引用:從來(lái)不會(huì)被回收
- 軟引用:當(dāng)內(nèi)存不足時(shí)會(huì)被回收
- 弱引用:正常垃圾回收時(shí)回收
- 虛引用:任何時(shí)刻都會(huì)被垃圾回收器回收
關(guān)注公眾號(hào):知識(shí)追尋者 獲取原創(chuàng)PDF,面試題
總結(jié)
以上是生活随笔為你收集整理的的引用_java中的强引用,软引用,弱引用,虚引用的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 设置横坐标刻度_是不是快被Excel的时
- 下一篇: python卡尔曼滤波跟踪_使用卡尔曼滤