溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

JVM中內存區域的作用是什么

發布時間:2021-08-02 17:24:29 來源:億速云 閱讀:160 作者:Leah 欄目:編程語言

JVM中內存區域的作用是什么,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

什么是jvm內存區域劃分?

其實這個問題非常簡單,JVM在運行我們寫好的代碼時,他是必須使用多塊內存空間的,不同的內存空間用來放不同的數據,然后配合我們寫的代碼流程,才能讓我們的系統運行起來。

舉個最簡單的例子,比如咱們現在知道了JVM會加載類到內存里來供后續運行,那么我問問大家,這些類加載到內存以后,放到哪兒去了呢?想過這個問題嗎?

所以JVM里就必須有一塊內存區域,用來存放我們寫的那些類。

包括我們定義的成員變量,類變量,方法,局部變量等等,都在jvm內存中對應著一塊內存來記錄存儲。

存放類的方法區

在JDK1.8之前的版本里,代表JVM的一塊區域。在1.8版本以后,這塊區域的名字改了,叫做“Matespace”,可以認為是“元數據空間”這樣的意思,當然這里主要存放的還是我們自己寫的各種類的相關信息。

舉個栗子。有如下兩個類,People類沒有成員變量,而Student類有一個name的類變量。

public class Student{  private static String name = "lisi";}public class People{  public static void main(){    Student student = new Student();  }}

這兩個類被加載到JVM,就會存放在這個方法區里面(注意:如果讀過我之前的章節,就會明白這里的加載代表的是:加載->驗證->準備->解析->初始化,類的所有類變量都會被賦值)。

執行代碼指令的程序計數器

我們知道,被加載到jvm的類對象是我們寫的.java文件被編譯之后的.class文件。

在編譯過后會將我們的代碼編譯成計算機能讀懂的字節碼。而這個.calss文件就是,就是我們代碼編譯好的字節碼了。

加載到內存以后,字節碼執行引擎就開始工作了。去執行我們編譯出來的代碼指令,如下圖

此時問題來了,我們是不是需要一塊內存空間來記錄我們字節碼執行引擎目前執行到了哪行代碼?這一塊特殊的內存區域就是“程序計數器”

這個程序計數器就是用來記錄當前執行的字節碼指令的位置。

如下圖:

到這里我相信會有人產生疑惑,就按照當前的代碼順序執行就行了,為什么要記錄執行到哪里了?

因為我們寫好的代碼可能會開啟多個線程并發的執行不同的代碼??赡墚斍熬€程這段代碼還沒有執行完畢,就上下文切換到另一段代碼中。

當線程再次上下文切換到之前的代碼時,就需要一個專門記錄當前線程執行到了哪一條字節碼。所以,每一個線程都有這自己的程序計數器。

如下圖:

java虛擬機棧

java代碼在執行的時候,一定是某個線程來執行某個方法中的代碼。

當線程執行到某個方法的時候,如果這個方法有局部變量,那么就需要一塊區域來存放局部變量的數據信息。這個區域就叫做java虛擬機棧。

每一個線程都有一個自己的java虛擬機棧,比如說當執行main方法的時候就會有一個main線程,用來存放main方法中定義的局部變量

public static void main(){  People people = new People();  int i = 9;}

比如上面的main()方法中,其實就有一個"people"的局部變量,他是引用一個People的實例對象的,這個對象我們先不管他。然后有一個"i"的局部變量。

如下圖:

我想大家應該都知道棧的數據結構,后進先出。當方法執行完畢以后,這個棧楨就會出棧,里面的局部變量信息就會從內存刪除。所以局部變量是線程安全的。因為只有當前線程能獲取到這個值。

為什么要用后進先出的數據結構?

假設a方法當中同步調用b方法,此時a方法的棧楨先入棧,然后再是b方法的棧楨入棧。b方法執行完畢后,b方法的棧楨出棧,繼續執行a方法。所以使用一個后進先出的棧結構是非常完美的。

此時jvm的內存模型圖如下:

java堆內存

這一塊內存是非常非常重要的。

我們實例化的所有對象都是存放在這個內存中。這個實例化的對象里面會包含一些數據,我們用上面的代碼來做栗子。

public class Student{  private String name = "lisi";  public String getNmae(){    return name;  }}public class People{  public static void main(){    Student student = new Student();    student.getName();  }}

還是這個代碼,當main線程執行main()方法的時候,首先在堆內存中實例化Student對象,然后在局部變量中創建student,student存的是實例化Student對象的內存地址。然后執行Student對象的getName()方法。

如下圖:

由上圖可以看出來,??臻g是封閉的,是線程安全的,而堆內存中是我們主要發生線程不安全的地方,因為堆內存的空間所有的線程其實都是能共享的。

此時jvm的內存劃分的最終模型為:

其他內存區域

很多java程序猿對這一塊區域的接觸是非常少的。

其實在JDK的很多底層代碼API中,比如NIO。

如果你去看源碼會發現很多地方的代碼不是java寫的,而是走的native方法去調用本地操作系統里面的一些方法,可能調用的都是c語言寫的方法。

比如說:public native int hashCode();

在調用這種native方法的時候,就會有線程對應的本地方法棧,這個其實類似于java虛擬機棧。也是存放各種native方法的局部變量表之類的信息。

還有一塊區域,是不是jvm的,通過NIO中的allocateDirect這種API,可以在jva堆外分配內存空間,然后通過java虛擬機棧里的DirectByteBuffer來引用和操作堆外內存空間。

看完上述內容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注億速云行業資訊頻道,感謝您對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

jvm
AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女