你不可不知的Java引用類型之——強引用


定義

強引用是使用最普遍的引用。如果一個對象具有強引用,那垃圾回收器寧願拋出OOM(OutOfMemoryError)也不會回收它。

說明

不要被這個強字嚇到,以為這個引用就很厲害,其實強引用就是程序中使用的一般引用類型。舉個簡單的栗子:

String s = new String("Hello Frank!");

強可達
如果一個對象與GC Roots之間存在強引用,則稱這個對象為強可達(strong reachable)對象。

關於可達性如果不太清楚的話,可以翻閱這篇文章

當你聲明一個變量並指向一個實例的時候,其實就是在創造一個強引用。那么,既然叫強引用,它“強”在哪里呢?

這主要體現在JVM進行GC的時候,只要對象有強引用與其關聯,就絕對不會對它進行回收,即使已經內存不足了也不會收回有強引用指向的對象。

如果你不需要使用某個對象了,可以將相應的引用設置為null,消除強引用來幫助垃圾回收器進行回收。因為過多的強引用也是導致OOM的罪魁禍首。

s = null;

顯式地設置消除引用,或已超出對象的生命周期范圍,則JVM會認為該對象不存在引用,這時就可能會回收這個對象。但是具體什么時候收集這要取決於具體的GC算法。

如果在一個方法的內部有一個變量s持有一個對象(Object)的強引用,那么這個變量s保存在棧中,而真正的引用內容(object)保存在堆中。當這個方法運行完成后就會退出方法棧,則引用s也會被銷毀,這個object就會被回收。但是當這個s是全局變量時,就需要在不再使用這個對象時賦值為null,因為有強引用關聯的對象是不會被垃圾回收的。

下面看另一個🌰:

A a = new A();
B b = new B(a);
a = null;

這里a和b是強引用,當把 a = null 時,這時 a 不再指向 A 的地址。講道理:當某個對象不再與其他引用關聯時,就會被 垃圾回收器判定為可回收,在GC中就會被回收掉。但是這里a = null 時,A 對象不能被回收,因為還有一個B對象持有其強引用,這時候就造成了內存泄漏

再看另一個會導致內存泄漏的栗子:

public static ArrayList<Object> list = new ArrayList<Object>();
public void stackOverflowTest(Object object){
    list.add(object);
    object = null;
}

GC回收的是不可達、弱可達或者虛可達對象,但是,在這個靜態集合類對象中,持有了對象的強引用,但是卻有可能對象已經不再使用了,所以當非靜態對象被靜態變量持有強引用的時候,最容易發生內存泄露,在方法中從list獲取到對象后賦值給一個變量,使用完之后將這個變量設置為null並不會釋放object引用的對象,因為list中還是持有對象的強引用。這時就造成了內存泄漏

小結

所以小結一下強引用的特點:

  • 強引用就是最普通的引用
  • 可以使用強引用直接訪問目標對象
  • 強引用指向的對象在任何時候都不會被系統回收
  • 強引用可能會導致內存泄漏
  • 過多的強引用會導致OOM


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM