自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java內(nèi)存泄漏最全詳解(六大原因及解決方案)

數(shù)據(jù)庫 其他數(shù)據(jù)庫
在對數(shù)據(jù)庫進(jìn)行操作的過程中,首先需要建立與數(shù)據(jù)庫的連接,當(dāng)不再使用時,需要調(diào)用close方法來釋放與數(shù)據(jù)庫的連接。

內(nèi)存泄漏的原因

JVM 虛擬機(jī)是使用引用計數(shù)法和可達(dá)性分析來判斷對象是否可回收,本質(zhì)是判斷一個對象是否還被引用,如果沒有引用則回收。

在開發(fā)的過程中,由于代碼的實現(xiàn)不同就會出現(xiàn)很多種內(nèi)存泄漏問題,讓gc 系統(tǒng)誤以為此對象還在引用中,無法回收,造成內(nèi)存泄漏。

內(nèi)存泄漏的危害

  • 長時間運行,程序變卡,性能嚴(yán)重下降
  • 程序莫名其妙掛掉
  • OutOfMemoryError錯誤
  • 亂七八糟的錯誤,還不易排查

內(nèi)存泄漏有哪些情況

內(nèi)存泄漏原因很多,因此這里給出最常見的幾種。

1.資源未關(guān)閉造成的內(nèi)存泄漏

各種連接,比如數(shù)據(jù)庫連接、網(wǎng)絡(luò)連接和IO連接等。

在對數(shù)據(jù)庫進(jìn)行操作的過程中,首先需要建立與數(shù)據(jù)庫的連接,當(dāng)不再使用時,需要調(diào)用close方法來釋放與數(shù)據(jù)庫的連接。

只有連接被關(guān)閉后,垃圾回收器才會回收對應(yīng)的對象。否則,如果在訪問數(shù)據(jù)庫的過程中,對Connection、Statement或ResultSet不顯性地關(guān)閉,將會造成大量的對象無法被回收,從而引起內(nèi)存泄漏,因此最好按照下面的做法處理資源類,偽代碼如下:

publicvoidhandleResource {
try {
// open connection
// handle business
} catch (Throwable t) {
// log stack
} finally {
// close connection
}}

2.靜態(tài)集合類

如HashMap、LinkedList等等,如果這些容器為靜態(tài)的,那么它們的生命周期與程序一致,則容器中的對象在程序結(jié)束之前將不能被釋放,從而造成內(nèi)存泄漏。

生命周期長的對象持有短生命周期對象的引用,盡管短生命周期的對象不再使用,但是因為長生命周期對象持有它的引用而導(dǎo)致不能被回收。

3.ThreadLocal的誤用

ThreadLocal一定要列在Java內(nèi)存泄露的榜首,總能在不知不覺中將內(nèi)存泄露掉,一個常見的例子是:

publicvoidtestThreadLocalMemoryLeaks {
ThreadLocal<List<Integer>> localCache = new ThreadLocal<>;
List<Integer> cacheInstance = new ArrayList<>(10000);
localCache.set(cacheInstance);localCache = new ThreadLocal<>;
}

當(dāng)localCache的值被重置之后cacheInstance被ThreadLocalMap中的value引用,無法被GC。

但是其key對ThreadLocal實例的引用是一個弱引用,本來ThreadLocal的實例被localCache和ThreadLocalMap的key同時引用,但是當(dāng)localCache的引用被重置之后。

則ThreadLocal的實例只有ThreadLocalMap的key這樣一個弱引用了,此時這個實例在GC的時候能夠被清理。

圖片

上面這張圖詳細(xì)地揭示了ThreadLocal和Thread以及ThreadLocalMap三者的關(guān)系。

1)Thread中有一個map,就是ThreadLocalMap

2)ThreadLocalMap的key是ThreadLocal,值是我們自己設(shè)定的。

3)ThreadLocal是一個弱引用,當(dāng)為null時,會被當(dāng)成垃圾回收

重點來了,突然我們ThreadLocal是null了,也就是要被垃圾回收器回收了,但是此時我們的ThreadLocalMap生命周期和Thread的一樣,它不會回收,這時候就出現(xiàn)了一個現(xiàn)象。

那就是ThreadLocalMap的key沒了,但是value還在,這就造成了內(nèi)存泄漏。

解決辦法:使用完ThreadLocal后,執(zhí)行remove操作,避免出現(xiàn)內(nèi)存溢出情況。

4.變量不合理的作用域

一般而言,一個變量的定義的作用范圍大于其使用范圍,很有可能會造成內(nèi)存泄漏。另一方面,如果沒有及時地把對象設(shè)置為null,很有可能導(dǎo)致內(nèi)存泄漏的發(fā)生。

public class UsingRandom {
private String msg;
public void receiveMsg(){
readFromNet();// 從網(wǎng)絡(luò)中接受數(shù)據(jù)保存到msg中
saveDB();// 把msg保存到數(shù)據(jù)庫中
}
}

如上面這個偽代碼,通過readFromNet方法把接受的消息保存在變量msg中,然后調(diào)用saveDB方法把msg的內(nèi)容保存到數(shù)據(jù)庫中,此時msg已經(jīng)就沒用了,由于msg的生命周期與對象的生命周期相同,此時msg還不能回收,因此造成了內(nèi)存泄漏。

實際上這個msg變量可以放在receiveMsg方法內(nèi)部,當(dāng)方法使用完,那么msg的生命周期也就結(jié)束,此時就可以回收了。還有一種方法,在使用完msg后,把msg設(shè)置為null,這樣垃圾回收器也會回收msg的內(nèi)存空間。

5.內(nèi)部類持有外部類

如果一個外部類的實例對象的方法返回了一個內(nèi)部類的實例對象,這個內(nèi)部類對象被長期引用了,即使那個外部類實例對象不再被使用,但由于內(nèi)部類持有外部類的實例對象,這個外部類對象將不會被垃圾回收,這也會造成內(nèi)存泄露。

6.堆外內(nèi)存無法回收

堆外內(nèi)存不受gc的管理,可能因為第三方的bug出現(xiàn)內(nèi)存泄漏

內(nèi)存泄漏的解決辦法

1.少使用靜態(tài)變量

盡量減少使用靜態(tài)變量,或者使用完及時 賦值為 null;

2.明確對象有效作用域

明確內(nèi)存對象的有效作用域,盡量縮小對象的作用域,能用局部變量處理的不用成員變量,因為局部變量彈棧會自動回收;

3.注意聲明周期引用

減少長生命周期的對象持有短生命周期的引用;

4.注意Sting的使用

使用StringBuilder和StringBuffer進(jìn)行字符串連接,Sting和StringBuilder以及StringBuffer等都可以代表字符串,其中String字符串代表的是不可變的字符串,后兩者表示可變的字符串。

如果使用多個String對象進(jìn)行字符串連接運算,在運行時可能產(chǎn)生大量臨時字符串,這些字符串會保存在內(nèi)存中從而導(dǎo)致程序性能下降。

5.不需要對象手動設(shè)置Null

對于不需要使用的對象手動設(shè)置null值,不管GC何時會開始清理,我們都應(yīng)及時的將無用的對象標(biāo)記為可被清理的對象;

6.及時關(guān)閉各種鏈接

各種連接(數(shù)據(jù)庫連接,網(wǎng)絡(luò)連接,IO連接)操作,務(wù)必顯示調(diào)用close關(guān)閉。

責(zé)任編輯:武曉燕 來源: mikechen的互聯(lián)網(wǎng)架構(gòu)
相關(guān)推薦

2016-12-15 21:47:11

Android內(nèi)存泄漏

2016-08-22 08:36:14

ReactiveCoc內(nèi)存泄漏GitHub

2020-06-17 07:00:00

Java數(shù)據(jù)科學(xué)家

2018-05-30 08:10:34

智慧農(nóng)業(yè)物聯(lián)網(wǎng)物聯(lián)網(wǎng)應(yīng)用

2023-02-06 10:37:50

數(shù)據(jù)驅(qū)動IT領(lǐng)導(dǎo)者

2014-12-02 09:57:41

Node.js

2014-12-03 10:14:11

Node.js

2017-08-08 16:35:26

Python爆紅原因

2022-07-21 15:41:30

OceanStor

2016-11-29 16:29:25

國產(chǎn)存儲失敗

2013-08-12 09:51:23

周鴻祎互聯(lián)網(wǎng)

2023-10-13 08:12:27

2018-10-12 14:34:13

2014-03-18 10:56:40

2009-07-06 09:16:30

ERP人才流失

2011-01-04 09:20:00

2013-08-07 10:16:43

Android內(nèi)存泄漏

2009-01-11 09:23:00

2023-08-28 14:13:08

2010-09-09 13:54:06

DIV CSS
點贊
收藏

51CTO技術(shù)棧公眾號