成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

淺析JVM內存結構和6大區域

開發 后端
內存作為系統中重要的資源,對于系統穩定運行和高效運行起到了關鍵的作用,Java和C之類的語言不同,不需要開發人員來分配內存和回收內存,而是由JVM來管理對象內存的分配以及對象內存的回收(又稱為垃圾回收、GC),這對于開發人員來說確實大大降低了編寫程序的難度,但帶來的一個副作用就是,當系統運行過程中出現JVM拋出的內存異常(例如OutOfMemoryError)的時候,很難知道原因是什么,另外一方面,要編寫高性能的程序,通常需要借助內存來提升性能,因此如何才能合理的使用內存以及讓JVM合理的進行內存的回收是必須掌握的,本文將主

 其實對于我們一般理解的計算機內存,它算是CPU與計算機打交道最頻繁的區域,所有數據都是先經過硬盤至內存,然后由CPU再從內存中獲取數據進行處理,又將數據保存到內存,通過分頁或分片技術將內存中的數據再flush至硬盤。那JVM的內存結構到底是如何呢?JVM做為一個運行在操作系統上,但又獨立于os運行的平臺,它的內存至少應該包括象寄存器、堆棧等區域。

JVM在運行時將數據劃分為了6個區域來存儲,而不僅僅是大家熟知的Heap區域,這6個區域圖示如下:

 JVM內存的分配結構示意圖

下面將逐一介紹下各個區域所做的工作及其充當的功能。

PC Register(PC寄存器)

PC寄存器是一塊很小的內存區域,主要作用是記錄當前線程所執行的字節碼的行號。字節碼解釋器工作時就是通過改變當前線程的程序計數器選取下一條字節碼指令來工作的。任何分支,循環,方法調用,判斷,異常處理,線程等待以及恢復線程,遞歸等等都是通過這個計數器來完成的。

 由于Java多線程是通過交替線程輪流切換并分配處理器時間的方式來實現的,在任何一個確定的時間里,在處理器的一個內核只會執行一條線程中的指令。因此為了線程等待結束需要恢復到正確的位置執行,每條線程都會有一個獨立的程序計數器來記錄當前指令的行號。計數器之間相互獨立互不影響,我們稱這塊內存為“線程私有”的內存。

如果所調用的方法為native的,則PC寄存器中不存儲任何信息。

l  JVM棧

JVM棧是線程私有的,每個線程創建的同時都會創建JVM棧,JVM棧中存放的為當前線程中局部基本類型的變量(java中定義的八種基本類型:boolean、char、byte、short、int、long、float、double)、部分的返回結果以及Stack Frame,非基本類型的對象在JVM棧上僅存放一個指向堆上的地址,因此Java中基本類型的變量是值傳遞,而非基本類型的變量是引用傳遞,Sun           JDK的實現中JVM棧的空間是在物理內存上分配的,而不是從堆上分配。

由于JVM棧是線程私有的,因此其在內存分配上非常高效,并且當線程運行完畢后,這些內存也就被自動回收。

當JVM棧的空間不足時,會拋出StackOverflowError的錯誤,在Sun JDK中可以通過-Xss來指定棧的大小,例如如下代碼:

  1. new Thread(new Runnable(){ 
  2.            public void run() { 
  3.               loop(0); 
  4.            }        
  5.            private void loop (int i){ 
  6.               if(i!=1000){ 
  7.                   i++; 
  8. loop (i); 
  9.               } 
  10.               else
  11.                   return
  12.               } 
  13.            } 
  14.           }).start(); 

當JVM參數設置為-Xss1K,運行后會報出類似下面的錯誤:

Exception in thread "Thread-0"java.lang.StackOverflowError

l  堆(Heap)

Heap是大家最為熟悉的區域,它是JVM用來存儲對象實例以及數組值的區域,可以認為Java中所有通過new創建的對象的內存都在此分配,Heap中的對象的內存需要等待GC進行回收,Heap在32位的操作系統上***為2G,在64位的操作系統上則沒有限制,其大小通過-Xms和-Xmx來控制,-Xms為JVM啟動時申請的最小Heap內存,默認為物理內存的1/64但小于1G,-Xmx為JVM可申請的***Heap內存,默認為物理內存的1/4,默認當空余堆內存小于40%時,JVM會增大Heap的大小到-Xmx指定的大小,可通過-XX:MinHeapFreeRatio=來指定這個比例,當空余堆內存大于70%時,JVM會將Heap的大小往-Xms指定的大小調整,可通過-XX:MaxHeapFreeRatio=來指定這個比例,但對于運行系統而言,為了避免頻繁的Heap Size的大小,通常都會將-Xms和-Xmx的值設成一樣,因此這兩個用于調整比例的參數通常是沒用的。其實jvm中對于堆內存的分配、使用、管理、收集等有更為精巧的設計,具體可以在JVM堆內存分析中進行詳細介紹。

當堆中需要使用的內存超過其允許的大小時,會拋出OutOfMemory的錯誤信息。

l  方法區域(MethodArea)

方法區域存放了所加載的類的信息(名稱、修飾符等)、類中的靜態變量、類中定義為final類型的常量、類中的Field信息、類中的方法信息,當開發人員在程序中通過Class對象中的getName、isInterface等方法來獲取信息時,這些數據都來源于方法區域,可見方法區域的重要性。同樣,方法區域也是全局共享的,它在虛擬機啟動時在一定的條件下它也會被GC,當方法區域需要使用的內存超過其允許的大小時,會拋出OutOfMemory的錯誤信息。

在Sun JDK中這塊區域對應的為PermanetGeneration,又稱為持久代,默認為64M,可通過-XX:PermSize以及-XX:MaxPermSize來指定其大小。

l  運行時常量池(RuntimeConstant Pool)

類似C中的符號表,存放的為類中的固定的常量信息、方法和Field的引用信息等,其空間從方法區域中分配。類或接口的常量池在該類的class文件被java虛擬機成功裝載時分配。

l  本地方法堆棧(NativeMethod Stacks)

JVM采用本地方法堆棧來支持native方法的執行,此區域用于存儲每個native方法調用的狀態。

例如有這么一段代碼:

  1. public class A { 
  2.                    public static void main(String[]args){ 
  3.            String a="a"
  4.           String b="b"
  5.            String ab="ab"
  6.            System.out.println((a+b)==ab);       // false 
  7.            System.out.println(("a"+"b")==ab);   // true 
  8.            final String afinal="a"
  9.            String result=afinal+"b"
  10.            System.out.println(result==ab);      // true 
  11.            String plus=a+"b"
  12.            System.out.println(plus==ab);        // false 
  13.              System.out.println(plus.intern()==ab);  // true 
  14.     } 

 

分析下上面代碼執行的結果,可通過javap –verbose A來輔助理解分析。

l  (a+b)==ab

a+b是兩個變量相加,需要到運行時才能確定其值,到運行時后JVM會為兩者相加后產生一個新的對象,因此a+b==ab的結果為false。

l  (“a”+”b”)==ab

“a”+”b”是常量,在編譯時JVM已經將其變為”ab”字符串了,而ab=”ab”也是常量,這兩者在常量池即為同一地址,因此(“a”+”b”)==ab為true。

l  result==ab

result=afinal+”b”,afinal是個final的變量, result在編譯時也已經被轉變為了”ab”,和”ab”在常量池中同樣為同一地址,因此result==ab為true。

l  plus=ab

plus和a+b的情況是相同的,因此plus==ab為false。

l  plus.intern()==ab

這里的不同點在于調用了plus.intern()方法,這個方法的作用是獲取plus指向的常量池地址,因此plus.intern()==ab為true。

在掌握了JVM對象內存分配的機制后,接下來看看JVM是如何做到自動的對象內存回收的,這里指的的是Heap以及Method Area的回收,其他幾個區域的回收都由JVM簡單的按生命周期來進行管理

原文鏈接:http://blog.csdn.net/zhaozheng7758/article/details/8623562

 【編輯推薦】

責任編輯:彭凡 來源: zhaozheng7758的博客
相關推薦

2017-09-20 08:48:09

JVM內存結構

2010-09-25 12:54:24

JVM內存

2023-08-24 07:46:21

服務器JVM

2010-09-27 13:48:41

JVM內存結構

2015-12-28 11:41:57

JVM內存區域內存溢出

2021-11-26 00:00:48

JVM內存區域

2024-11-15 09:14:23

JDK4NIO函數

2010-09-27 09:31:42

JVM內存結構

2020-08-10 17:49:25

JVM內存溢出

2019-07-16 14:59:00

JVM內存區域

2020-03-30 11:10:34

JVM內存結構

2024-11-13 11:12:08

JVM內存區域

2024-08-27 10:34:05

2021-08-13 08:15:23

JVM 虛擬機Java

2021-06-30 10:29:59

中國電信ROADM網絡

2020-10-28 09:50:33

SpringBootJava

2010-09-16 15:24:16

JVM結構

2010-09-26 16:42:04

JVM內存組成JVM垃圾回收

2018-07-04 14:43:55

對象模型內存結構內存模型

2011-04-13 16:50:54

CC++內存
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 成人一区二区三区 | 国产视频h | 日韩aⅴ视频| 最新国产精品视频 | 成人一区二区三区在线观看 | 欧美激情综合 | 色综合久久久 | 毛片在线看片 | 日韩中文视频 | 欧美激情a∨在线视频播放 成人免费共享视频 | 男人天堂免费在线 | 成人精品免费视频 | 国产视频线观看永久免费 | 华丽的挑战在线观看 | 国产精品一级 | 中文字幕精品一区 | 精品久久精品 | 中文字幕日韩欧美一区二区三区 | 久久精品无码一区二区三区 | 精品一区二区三区在线视频 | 久久久国产一区 | 成人网在线 | 亚洲综合久久久 | 99pao成人国产永久免费视频 | 黄色免费在线观看网站 | 成人影院网站ww555久久精品 | 欧美大片一区 | 欧美精品在线一区 | 欧美久久久久久久久中文字幕 | 成人在线视频一区二区三区 | 国产农村一级国产农村 | 国产在线精品一区二区三区 | 国产欧美在线一区二区 | 日韩欧美国产一区二区 | 欧美一区二区三区在线 | 成人免费区一区二区三区 | 天天干狠狠操 | 久久久免费在线观看 | 视频1区2区 | 欧区一欧区二欧区三免费 | 久久国产精品亚洲 |