Java運行時數(shù)據(jù)區(qū)域(內(nèi)存劃分)的深入講解
1. 程序計數(shù)器(線程私有)
程序計數(shù)器是一塊比較小的內(nèi)存空間,可以看做是當(dāng)前線程所執(zhí)行的字節(jié)碼的行號指示器(切換線程后,能恢復(fù)到正確的執(zhí)行位置).
2. Java虛擬機棧(線程私有)
(1) 概念
虛擬機棧描述的是Java方法執(zhí)行的內(nèi)存模型 : 每個方法執(zhí)行的同時都會創(chuàng)建一個棧幀用于存儲局部變量表、操作數(shù)棧、動態(tài)鏈接、方法出口 等信息。每一個方法從調(diào)用直至執(zhí)行完成的過程,就對應(yīng)一個棧幀在虛擬機棧中入棧和出棧的過程。聲明周期與線程相同。
關(guān)于棧幀的理解: 創(chuàng)建啟動一個線程,就創(chuàng)建了一個虛擬機??臻g(存放多個節(jié)點,先進后出).可能出現(xiàn)兩種異常: ①StackOverflowError: 方法調(diào)用鏈太深(如遞歸); ②OOM
(2) 下面我們來分析一段代碼
public class VMStackTest {
public static void main(String[] args) {
int m = 1;
int n = 2;
swap1(1, 2);
System.out.printf("main:m=%s, n=%s%n", m, n);
}
private static void swap1(int m, int n) {
int tmp = m;
m = n;
n = tmp;
Person p1 = new Person("p1");
Person p2 = new Person("p2");
swap2(p1, p2);
System.out.printf("swap1:p1=%s, p2=%s%n", p1.name, p2.name);
}
private static void swap2(Person p1, Person p2) {
Person tmp = p1;
p1 = p2;
p2 = tmp;
System.out.printf("swap2:p1=%s, p2=%s%n", p1.name, p2.name);
}
static class Person{
String name;
public Person(String name) {
this.name = name;
}
}
}
這段代碼的執(zhí)行結(jié)果是什么?
我們知道,Java中只有值傳遞,當(dāng)為基本數(shù)據(jù)類型時,傳的是字面值常量,當(dāng)為引用類型,傳的是地址.
swap1中,只是修改了方法棧幀中的局部變量,對外面的局部變量不起作用,所以swap1中的交換是無效的.
swap2中也一樣,只是修改了方法棧幀中的局部變量,對外面的局部變量交換也是無效的.
所以最終的打印結(jié)果為:

如果要把p1,p2對象的名稱調(diào)換,p1.name="p2", p2.name="p1"(作用在堆里邊的對象)
3. 本地方法棧(線程私有)
本地方法棧與虛擬機棧的作用完全一樣,他倆的區(qū)別無非是本地方法棧為虛擬機使用的Native方法服務(wù),而虛擬機棧為JVM執(zhí)行的Java方法服務(wù)。
4. Java堆(線程共享)
Java堆(Java Heap)是JVM所管理的最大內(nèi)存區(qū)域。Java堆是所有線程共享的一塊區(qū)域,在JVM啟動時創(chuàng)建。此內(nèi)存區(qū)域存放的都是對象實例。JVM規(guī)范中說到:"所有的對象實例以及數(shù)組都要在堆上分配"。
5. 方法區(qū)(線程共享)
方法區(qū)與Java堆一樣,是各個線程共享的內(nèi)存區(qū)域。它用于存儲已被虛擬機加載的類信息、常量、靜態(tài)變量(jdk1.7在方法區(qū)中;1.8在堆中)、即時編譯器編譯后的代碼等數(shù)據(jù)。
6. 運行時常量池(方法區(qū)的一部分)
(1) 概念
運行時常量池是方法區(qū)的一部分,存放字面量與符號引用。
字面量 : 字符串(JDK1.7后移動到堆中) 、final常量、基本數(shù)據(jù)類型的值。
符號引用 : 類和結(jié)構(gòu)的完全限定名、字段的名稱和描述符、方法的名稱和描述符。
(2) 補充: 其他常量池
① class文件常量池: Java文件編譯為class字節(jié)碼文件,存在的常量池,包含字面量和符號引用.
② Java進程運行后,即使沒有執(zhí)行到某行代碼,也已經(jīng)把class文件常量池中的內(nèi)容放在運行時常量池,如下圖:

③ 字符串常量池: 1.7之前是在運行時常量池里面,1.7之后,是在堆里面.
總結(jié)
到此這篇關(guān)于Java運行時數(shù)據(jù)區(qū)域(內(nèi)存劃分)的文章就介紹到這了,更多相關(guān)Java運行時內(nèi)存劃分內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
IntelliJ IDEA修改新建文件自動生成注釋的user名
今天小編就為大家分享一篇關(guān)于IntelliJ IDEA修改新建文件自動生成注釋的user名,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2018-10-10

