Java 8下PermGen及參數設置
在上一章節中,我們定制化使用了Java 8環境,使用我們的測試頁面打印出了JVM基本參數,但如果我們自己觀察,會發現在MXBeans中,沒有出現PermGen的使用數據,初始大小等信息,即使我們已經設置了大小:
在Java 7及以前版本中,PermGen主要存放加載的類和元數據信息,如果設置過小,類加載失敗,可能會出現OutOfMemory的經典錯誤,在Azure website里面的Java開發中,碰到的客戶定制化設置的問題也會涉及到PermGen的大小定制化。
那么在Java 8里面,PermGen去哪了呢?
我們可以先來看一下JVM的內存模型,JVM的內存分為Heap memory和Non-Heap memory,Heap memory主要會存放一些Java Object對象信息,而non-heap memory如PermGen主要會存放一些加載的Java classes和元數據信息。
在Java 8之前的Java版本當中,Heap size可以通過 MS, MX進行大小設置,而PermGen可以通過PermSize, MaxPerm在Java option中進行大小設置。之前的JVM的一個明顯問題是,你在啟動的時候設置了XX:MaxPermSize,那么一旦在運行過程中加載的類超過了這個大小限制,你就會馬上碰到那個著名的OOM(out of memory)錯誤,當然這種設計除了會出現OOM,也會有無法動態調整,很難調優等缺點,也會導致一系列的bug和性能問題,例如:
因此上,在Java 8的設計中,Oracle和Java社區放棄了PermGen設置,從此之后不再有PermGen這樣一個東西,但是元數據依然是需要保存的,所以在Java 8中,元數據移到了本地內存中,叫Metaspace的地方。
那么對于最終用戶來講有什么變化呢?你不會因為PermGen碰到OOM的問題,因為你所有的有效系統內存都可以做Metaspace了,所以你不需要單獨設置metaspace,在一個64位系統的機器中,默認的Metaspace初始大小是21MB,那么最大呢?理論上如果你的元數據真的非常多,加載的類也很多,理論上最大可以用光你所有的有效系統內存。
那么有沒有辦法設置或者限制Metaspace的大小呢?可以,Java提供了兩個參數來讓你在必要的情況下設置Metaspace的大小:
–XX:MetaspaceSize:64微系統默認21MB,你可以設置大一些避免頻繁的full GC
-XX:MaxMetaspaceSize:理論上大小沒有限制,但你可以設置一個限制值
回到我們的Azure website的Java OPTS的設置上,在Java 8的環境里面PermSize已經無用了,所以web.config中設置可簡化為: