溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java中什么情況會導(dǎo)致內(nèi)存泄漏

發(fā)布時間:2021-10-29 11:24:07 來源:億速云 閱讀:951 作者:iii 欄目:編程語言

這篇文章主要講解了“Java中什么情況會導(dǎo)致內(nèi)存泄漏”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Java中什么情況會導(dǎo)致內(nèi)存泄漏”吧!

概念

內(nèi)存泄露:指程序中動態(tài)分配內(nèi)存給一些臨時對象,但對象不會被GC回收,它始終占用內(nèi)存,被分配的對象可達(dá)但已無用。即無用對象持續(xù)占有內(nèi)存或無用對象的內(nèi)存得不到及時釋放,從而造成的內(nèi)存空間浪費(fèi)。

可達(dá)性分析算法

JVM使用可達(dá)性分析算法判斷對象是否存活。

GC Root

通過一系列名為“GC Roots”的對象作為起點,從這些結(jié)點開始向下搜索,搜索所走過的路徑稱為“引用鏈(Reference Chain)”,當(dāng)一個對象到GC Roots沒有任何飲用鏈相連時,則證明此對象是不可用的。

Java中什么情況會導(dǎo)致內(nèi)存泄漏

object4、object5、object6雖然有互相判斷,但是它們到GC Rootd是不可達(dá)的,所以它們將會判定為是可回收對象。

可以作為GC Roots的對象有:

  •  虛擬機(jī)棧(棧幀中的本地變量表)中的引用的對象;

  •  方法區(qū)中的類靜態(tài)屬性引用的對象;

  •  方法區(qū)中的常量引用的對象;

  •  本地方法棧中JNI的引用的對象

雖然Java有垃圾收集器幫組實現(xiàn)內(nèi)存自動管理,雖然GC有效的處理了大部分內(nèi)存,但是并不能完全保證內(nèi)存的不泄漏。

內(nèi)存泄漏

內(nèi)存泄漏就是堆內(nèi)存中不再使用的對象無法被垃圾收集器清除掉,因此它們會不必要地存在。這樣就導(dǎo)致了內(nèi)存消耗,降低了系統(tǒng)的性能,最終導(dǎo)致OOM使得進(jìn)程終止。

內(nèi)存泄漏的表現(xiàn):

  •  應(yīng)用程序長時間連續(xù)運(yùn)行時性能嚴(yán)重下降;

  •  應(yīng)用程序中的OutOfMemoryError堆錯誤;

  •  自發(fā)且奇怪的應(yīng)用程序崩潰;

  •  應(yīng)用程序偶爾會耗盡連接對象;

可能導(dǎo)致內(nèi)存泄漏的原因:

1. static字段引起的內(nèi)存泄漏

大量使用static字段會潛在的導(dǎo)致內(nèi)存泄漏,在Java中,靜態(tài)字段通常擁有與整個應(yīng)用程序相匹配的生命周期。

解決辦法:最大限度的減少靜態(tài)變量的使用;單例模式時,依賴于延遲加載對象而不是立即加載的方式(即采用懶漢模式,而不是餓漢模式)

2. 未關(guān)閉的資源導(dǎo)致內(nèi)存泄漏

每當(dāng)創(chuàng)建連接或者打開流時,JVM都會為這些資源分配內(nèi)存。如果沒有關(guān)閉連接,會導(dǎo)致持續(xù)占有內(nèi)存。在任意情況下,資源留下的開放連接都會消耗內(nèi)存,如果不處理,就會降低性能,甚至OOM。

解決辦法:使用finally塊關(guān)閉資源;關(guān)閉資源的代碼,不應(yīng)該有異常;JDK1.7之后,可以使用太try-with-resource塊。

3. 不正確的equals()和hashCode()

在HashMap和HashSet這種集合中,常常用到equal()和hashCode()來比較對象,如果重寫不合理,將會成為潛在的內(nèi)存泄漏問題。

解決辦法:用最佳的方式重寫equals()和hashCode().

4. 引用了外部類的內(nèi)部類

非靜態(tài)內(nèi)部類的初始化,總是需要外部類的實例;默認(rèn)情況下,每個非靜態(tài)內(nèi)部類都包含對其外部類的隱式引用,如果我們在應(yīng)用程序中使用這個內(nèi)部類對象,那么即使在我們的外部類對象超出范圍后,它也不會被垃圾收集器清除掉。

解決辦法:如果內(nèi)部類不需要訪問外部類包含的類成員,可以轉(zhuǎn)換為靜態(tài)類。

5. finalize方法導(dǎo)致的內(nèi)存泄漏

重寫finalize()方法時,該類的對象不會立即被垃圾收集器收集,如果finalize()方法的代碼有問題,那么會潛在的印發(fā)OOM;

解決辦法:避免重寫finalize()方法。

6. 常量字符串造成的內(nèi)存泄漏

如果我們讀取一個很大的String對象,并調(diào)用了intern(),那么它將放到字符串池中,位于PermGen中,只要應(yīng)用程序運(yùn)行,該字符串就會保留,這就會占用內(nèi)存,可能造成OOM。(針對JDK1.6及以前,常量池在PermGen永久代中)

解決辦法:增加PermGen的大小,-XX:MaxPermSize=512M;JDK1.7以后字符串池轉(zhuǎn)移到了堆中。

intern()方法詳解:

String str1 = "abc";  String str2 = "abc";  String str3 = new String("abc");  String str4 = str3.intern();  System.out.println(str1 == str2);  System.out.println(str2 == str3);  System.out.println(str1 == str4);  System.out.println(str3 == str4);  true, false, true, false

intern()方法搜索字符串常量池,如果存在指定的字符串,就返回之;

否則,就將該字符串放入常量池并返回之。

換言之,intern()方法保證每次返回的都是 同一個字符串對象

String str1 = "abc";  String str2 = "abc";  String str3 = new String("abcd");  String str4 = str3.intern();  String str5 = "abcd";  System.out.println(str1 == str2);  System.out.println(str2 == str3);  System.out.println(str1 == str4);  System.out.println(str3 == str4);  System.out.println(str4 == str5);  true  false  false  false  true

為何要使用intern()方法?看看equals方法的源碼:

public boolean equals(Object anObject) {      if (this == anObject) {          return true;      }      if (anObject instanceof String) {          String anotherString = (String)anObject;          int n = value.length;          if (n == anotherString.value.length) {              char v1[] = value;              char v2[] = anotherString.value;              int i = 0;              while (n-- != 0) {                  if (v1[i] != v2[i])                      return false;                  i++;              }              return true;          }      }      return false;  }

可以看到,比較兩個字符串的時候,首先比較兩個字符串對象是否地址相同,不同再挨個比較字符。這樣就大大加快了比較的速度。否則若每次都挨個比較將是非常耗時的。

7. 使用ThreadLocal造成內(nèi)存泄漏

使用ThreadLocal時,每個線程只要處于存活狀態(tài)就可保留對其ThreadLocal變量副本的隱式調(diào)用,且將保留其自己的副本。使用不當(dāng),就會引起內(nèi)存泄漏。

一旦線程不再存在,該線程的threadLocal對象就應(yīng)該被垃圾收集,而現(xiàn)在線程的創(chuàng)建都是使用線程池,線程池有線程重用的功能,因此線程就不會被垃圾回收器回收。所以使用到ThreadLocal來保留線程池中的線程的變量副本時,ThreadLocal沒有顯式地刪除時,就會一直保留在內(nèi)存中,不會被垃圾回收。

解決辦法:不再使用ThreadLocal時,調(diào)用remove()方法,該方法刪除了此變量的當(dāng)前線程值。不要使用ThreadLocal.set(null),它只是查找與當(dāng)前線程關(guān)聯(lián)的Map并將鍵值中這個threadLocal對象所對應(yīng)的值為null,并沒有清除這個鍵值對。

感謝各位的閱讀,以上就是“Java中什么情況會導(dǎo)致內(nèi)存泄漏”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Java中什么情況會導(dǎo)致內(nèi)存泄漏這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI