詳解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以上就是詳解JAVA 強引用的詳細內容,更多關于JAVA 強引用的資料請關注好吧啦網其它相關文章!
相關文章: