文字來自StackOverflow問答網站的一個熱門討論:如何用Java寫出一段會發生記憶體外洩的程式碼。
Q:剛才我參加了面試,面試官問我如何寫出會發生記憶體洩漏的Java程式碼。這個問題我一點想法都沒有,好囧。
A1:透過以下步驟可以輕鬆產生記憶體洩漏(程式碼不能存取到某些對象,但是它們仍然保存在記憶體中):
應用程式建立一個長時間運行的執行緒(或使用執行緒池,會更快地發生內存洩漏)。
執行緒透過某個類別載入器(可以自訂)載入一個類別。
該類別分配了大塊記憶體(例如new byte[1000000]),在某個靜態變數儲存一個強引用,然後在ThreadLocal中儲存它自身的引用。分配額外的記憶體new byte[1000000]是可選的(類別實例外洩已經足夠了),但是這樣會使記憶體外洩更快。
執行緒清理自訂的類別或載入該類別的類別載入器。
重複以上步驟。
由於沒有了對類別和類別載入器的引用,ThreadLocal中的儲存就不能被存取到。 ThreadLocal持有該物件的引用,它也就持有了這個類別及其類別載入器的引用,類別載入器持有它所載入的類別的所有引用,這樣GC無法回收ThreadLocal中儲存的記憶體。在許多JVM的實作中Java類別和類別載入器直接分配到permgen區域不執行GC,這導致了更嚴重的記憶體外洩。
這種洩露模式的變種之一就是如果你經常重新部署以任何形式使用了ThreadLocal的應用程式、應用容器(例如Tomcat)會很容易發生記憶體洩漏(由於應用容器使用瞭如前所述的線程,每次重新部署應用程式時將使用新的類別載入器)。
A2:
靜態變數參考物件
class MemorableClass { static final ArrayList list = new ArrayList(100); }
呼叫長字串的String.intern()
String str=readString(); // read lengthy string any source db,textbox/jsp etc.. // This will place the string in memory pool from which you cant remove str.intern();
未關閉已開啟流(文件,網路等)
reee達區域 例如透過native方法分配的記憶體。 web應用在application範圍的對象,應用未重啟或沒有明確移除 getServletContext().setAttribute("SOME_MAP", map);〜『移除 session.setAttribute("SOME_MAP", map); 不正確或不合適的JVM選項 例如IBM JDK的noclassgcc『未實作)hashCode()或equals(),會導致集合中持續增加「副本」。如果集合不能地忽略掉它應該忽略的元素,它的大小就只能持續成長,而且不能刪除這些元素。 如果你想要產生錯誤的鍵值對,可以像下面這樣做:try { BufferedReader br = new BufferedReader(new FileReader(inputFile)); ... ... } catch (Exception e) { e.printStacktrace(); }
使用InflaterInputStream在建構子(例如PNGImageDecoder)中傳遞new java.util.zip.Inflater(),不呼叫inflater的end()。只是new的話非常安全,但如果自己創建該類別作為構造函數參數時調用流的close()不能關閉inflater,可能發生內存洩漏。這並不是真正的記憶體洩漏因為它會被finalizer釋放。但這消耗了很多native內存,導致linux的oom_killer殺掉進程。所以這給我們的教訓是:盡可能提早釋放native資源。
java.util.zip.Deflater也是一樣,它的情況更加嚴重。好的地方可能是很少用到Deflater。如果自己創建了Deflater或Inflater記住必須呼叫end()。