1. 程式人生 > >JVM記憶體限制和調整

JVM記憶體限制和調整

堆(Heap)和非堆(Non-heap)記憶體
  按照官方的說法:“Java 虛擬機器具有一個堆,堆是執行時資料區域,所有類例項和陣列的記憶體均從此處分配。堆是在 Java 虛擬機器啟動時建立的。”“在JVM中堆之外的記憶體稱為非堆記憶體(Non-heap memory)”。可以看出JVM主要管理兩種型別的記憶體:堆和非堆。簡單來說堆就是Java程式碼可及的記憶體,是留給開發人員使用的;非堆就是JVM留給 自己用的,所以方法區、JVM內部處理或優化所需的記憶體(如JIT編譯後的程式碼快取)、每個類結構(如執行時常數池、欄位和方法資料)以及方法和構造方法 的程式碼都在非堆記憶體中。 

堆記憶體分配

  JVM初始分配的記憶體由-Xms指定,預設是實體記憶體的1/64;JVM最大分配的記憶體由-Xmx 指定,預設是實體記憶體的1/4。預設空餘堆記憶體小於40%時,JVM就會增大堆直到-Xmx的最大限制;空餘堆記憶體大於70%時,JVM會減少堆直到- Xms的最小限制。因此伺服器一般設定-Xms、-Xmx相等以避免在每次GC 後調整堆的大小。 
非堆記憶體分配
  JVM使用-XX:PermSize設定非堆記憶體初始值,預設是實體記憶體的1/64;由XX:MaxPermSize設定最大非堆記憶體的大小,預設是實體記憶體的1/4。 
JVM記憶體限制(最大值)
  首 先JVM記憶體限制於實際的最大實體記憶體(廢話!呵呵),假設實體記憶體無限大的話,JVM記憶體的最大值跟作業系統有很大的關係。簡單的說就32位處理器雖然 可控記憶體空間有4GB,但是具體的作業系統會給一個限制,這個限制一般是2GB-3GB(一般來說Windows系統下為1.5G-2G,Linux系統 下為2G-3G),而64bit以上的處理器就不會有限制了。 


  通常,我們為了避免記憶體溢位等問題,需要設定環境變數 
  JAVA_OPTS    -Xms256M -Xmx512M 等,【對於伺服器,一般都設定成一樣的】 
  但是有的時候可能這樣的設定還會不行(比如,當Server應用程式載入較多類時,即jvm載入類時,永久域中的物件急劇增加,從而使jvm不斷調整永久域大小,為了避免調整),你可以使用更多的引數配置, 
  如: java -Xms512m -Xmx512m -XX:PermSize=64m -XX:MaxPermSize=128m