1、Java中有哪幾種引用?它們的含義和區(qū)別是什么?
從JDK1.2開(kāi)始,Java中的引用類型分為四種,分別是:
①?gòu)?qiáng)引用(StrongReference) 不會(huì)被回收
②軟引用(SoftRefernce) 系統(tǒng)檢測(cè)內(nèi)存不足時(shí)會(huì)被回收
③弱引用(WeakReference) gc回收器觸發(fā)會(huì)被回收
④虛引用(PhantomReference)等于null
強(qiáng)引用-StrongReference
這種引用是平時(shí)開(kāi)發(fā)中最常用的,例如 String strong = new String("Strong Reference"),當(dāng)一個(gè)實(shí)例對(duì)象具有強(qiáng)引用時(shí),垃圾回收器不會(huì)回收該對(duì)象,當(dāng)內(nèi)存不足時(shí),寧愿拋出OutOfMemeryError異常也不會(huì)通過(guò)回收強(qiáng)引用的對(duì)象,因?yàn)镴VM認(rèn)為強(qiáng)引用的對(duì)象是用戶正在使用的對(duì)象,它無(wú)法分辨出到底該回收哪個(gè),強(qiáng)行回收有可能導(dǎo)致系統(tǒng)嚴(yán)重錯(cuò)誤。
軟引用-SoftReference
如果一個(gè)對(duì)象只有軟引用,那么只有當(dāng)內(nèi)存不足時(shí),JVM才會(huì)去回收該對(duì)象,其他情況不會(huì)回收。軟引用可以結(jié)合ReferenceQueue來(lái)使用,當(dāng)由于系統(tǒng)內(nèi)存不足,導(dǎo)致軟引用的對(duì)象被回收了,JVM會(huì)把這個(gè)軟引用加入到與之相關(guān)聯(lián)的ReferenceQueue中。
ReferenceQueue referenceQueue = new ReferenceQueue();
SoftReference<Book> softReference = new SoftReference<>(new Book(), referenceQueue);
Book book = softReference.get();
Reference reference = referenceQueue.poll();
當(dāng)系統(tǒng)內(nèi)存不足時(shí),觸發(fā)gc,這個(gè)Book就會(huì)被回收,reference 將不為null。
弱引用-WeakReference
只有弱引用的對(duì)象,當(dāng)JVM觸發(fā)gc時(shí),就會(huì)回收該對(duì)象。與軟引用不同的是,不管是否內(nèi)存不足,弱引用都會(huì)被回收。弱引用可以結(jié)合ReferenceQueue來(lái)使用,當(dāng)由于系統(tǒng)觸發(fā)gc,導(dǎo)致軟引用的對(duì)象被回收了,JVM會(huì)把這個(gè)弱引用加入到與之相關(guān)聯(lián)的ReferenceQueue中,不過(guò)由于垃圾收集器線程的優(yōu)先級(jí)很低,所以弱引用不一定會(huì)被很快回收。下面通過(guò)一個(gè)主動(dòng)觸發(fā)gc的例子來(lái)驗(yàn)證此結(jié)論。
ReferenceQueue referenceQueue = new ReferenceQueue();
WeakReference<Book> weakReference = new WeakReference(new Book(), referenceQueue);
Book book = softReference.get();
System.gc();
//Runtime.getRuntime().gc();
Reference reference = referenceQueue.poll();

當(dāng)然這不是每次都能復(fù)現(xiàn),因?yàn)槲覀冋{(diào)用System.gc()只是告訴JVM該回收垃圾了,但是它什么時(shí)候做還是不一定的,但就我測(cè)試來(lái)看,只要多寫(xiě)幾次System.gc(),復(fù)現(xiàn)的概率還是很高的。
虛引用-PhantomReference
如果一個(gè)對(duì)象只有虛引用在引用它,垃圾回收器是可以在任意時(shí)候?qū)ζ溥M(jìn)行回收的,虛引用主要用來(lái)跟蹤對(duì)象被垃圾回收器回收的活動(dòng),當(dāng)被回收時(shí),JVM會(huì)把這個(gè)弱引用加入到與之相關(guān)聯(lián)的ReferenceQueue中。與軟引用和弱引用不同的是,虛引用必須有一個(gè)與之關(guān)聯(lián)的ReferenceQueue,通過(guò)phantomReference.get()得到的值為null,試想一下,如果沒(méi)有ReferenceQueue與之關(guān)聯(lián)還有什么存在的價(jià)值呢?
PhantomReference<Book> phantomReference = new PhantomReference<>(new Book(), referenceQueue);
Book book = phantomReference.get(); //此值為null
Reference reference = referenceQueue.poll();