Java中堆、棧、常量池等概念解析

來源:互聯網
上載者:User
Java中堆、棧、常量池等概念解析

程式運行時,我們最好對資料儲存到什麼地方做到心中有數。特別要注意的是記憶體的分配。有六個地方都可以儲存資料:
(1) 寄存器。這是最快的儲存地區,因為它位於和其他所有儲存方式不同的地方:處理器內部。然而,寄存器的數量十分有限,所以寄存器是根據需要由編譯器分配。我們對此沒有直接的控制權,也不可能在自己的程式裡找到寄存器存在的任何蹤跡。

(2) 棧(stack)。存放基本類型的變數資料和對象的引用,但對象本身不存放在棧中,而是存放在堆(new 出來的對象)或者常量池中(字串常量對象存放在常量池中。) 駐留於常規RAM(隨機訪問儲存空間)地區,但可通過它的“堆棧指標”獲得處理的直接支援。堆棧指標若向下移,會建立新的記憶體;若向上移,則會釋放那些內 存。這是一種特別快、特別有效資料儲存方式,僅次於寄存器。建立程式時,Java編譯器必須準確地知道堆棧內儲存的所有資料的“長度”以及“存在時 間”。這是由於它必鬚生成相應的代碼,以便向上和向下移動指標。這一限制無疑影響了程式的靈活性,所以儘管有些Java資料要儲存在堆棧裡——特別是對象 控制代碼,但Java對象並不放到其中。

(3) 堆(heap)。存放所有new出來的對象,一種常規用途的記憶體池(也在RAM地區),其中保 存了Java對象。和堆棧不同,“記憶體堆”或“堆”(Heap)最迷人的地方在於編譯器不必知道要從堆裡分配多少儲存空間,也不必知道儲存的資料要在堆 裡停留多長的時間。因此,用堆儲存資料時會得到更大的靈活性。要求建立一個對象時,只需用new命令編製相關的代碼即可。執行這些代碼時,會在堆裡自動進 行資料的儲存。當然,為達到這種靈活性,必然會付出一定的代價:在堆裡分配儲存空間時會花掉更長的時間!

(4) 靜態儲存。存放靜態成員(static定義的),這兒的“靜態”(Static)是指“位於固定位置”(儘管也在RAM裡)。程式運行期間,靜態儲存的資料將隨時等候調用。可用static關鍵字指出一個對象的特定元素是靜態。但Java對象本身永遠都不會置入靜態儲存空間。

(5) 常數儲存。存放字串常量和基本類型常量(public static final)。 常數值通常直接置於程式碼內部。這樣做是安全的,因為它們永遠都不會改變。有的常數需要嚴格地保護,所以可考慮將它們置入唯讀記憶體(ROM)。

(6) 非RAM儲存。若資料完全獨立於一個程式之外,則程式不運行時仍可存在,並在程式的控制範圍之 外。其中兩個最主要的例子便是“流式對象”和“固定對象”。對於流式對象,對象會變成位元組流,通常會發給另一台機器。而對於固定對象,對象儲存在磁碟中。 即使程式中止運行,它們仍可保持自己的狀態不變。對於這些類型的資料存放區,一個特別有用的技巧就是它們能存在於其他媒體中。一旦需要,甚至能將它們恢複成 普通的、基於RAM的對象。Java 1.1提供了對Lightweight persistence的支援。未來的版本甚至可能提供更完整的方案

 

這裡我們主要關心棧,堆和常量池,對於棧和常量池中的對象可以共用,對於堆中的對象不可以共用。棧中的資料大小和生命週期是可以確定的,當沒有引用指向資料時,這個資料就會消失。堆中的對象的由記憶體回收行程負責回收,因此大小和生命週期不需要確定,具有很大的靈活性。
對於字串:其對象的引用都是儲存在棧中的,如果是編譯期已經建立好(直接用雙引號定義的)的就儲存在常量池中,如果是運行期(new出來的)才能確定的就儲存在堆中。對於equals相等的字串,在常量池中永遠只有一份,在堆中有多份。
如以下代碼:

Java代碼

String s1 = "china";  
String s2 = "china";  
String s3 = "china";  
String ss1 = new String("china");  
String ss2 = new String("china");  
String ss3 = new String("china");  
 

這裡解釋一下黃色這3個箭頭,對於通過new產生一個字串(假設為”china”)時,會先去常量池中尋找是否已經有了”china”對象,如果 沒有則在常量池中建立一個此字串對象,然後堆中再建立一個常量池中此”china”對象的拷貝對象。這也就是有道面試題:String s = new String(“xyz”);產生幾個對象?一個或兩個,如果常量池中原來沒有”xyz”,就是兩個。

對於基礎類型的變數和常量:變數和引用儲存在棧中,常量儲存在常量池中。
如以下代碼:

Java代碼

int i1 = 9;  
int i2 = 9;  
int i3 = 9;   
public static final int INT1 = 9;  
public static final int INT2 = 9;  
public static final int INT3 = 9;  

 
對於成員變數和局部變數:成員變數就是方法外部,類的內部定義的變數;局部變數就是方法或語句塊內部定義的變數。局部變數必須初始化。
形式參數是局部變數,局部變數的資料存在於棧記憶體中。棧記憶體中的局部變數隨著方法的消失而消失。
成員變數儲存在堆中的對象裡面,由記憶體回收行程負責回收。
如以下代碼:

Java代碼

class BirthDate {  
    private int day;  
    private int month;  
    private int year;      
    public BirthDate(int d, int m, int y) {  
        day = d;   
        month = m;   
        year = y;  
    }  
    省略get,set方法………  
}  

public class Test{  
    public static void main(String args[]){  
int date = 9;  
        Test test = new Test();        
           test.change(date);   
        BirthDate d1= new BirthDate(7,7,1970);         
    }    

    public void change1(int i){  
        i = 1234;  
    } 


 
對於以上這段代碼,date為局部變數,i,d,m,y都是形參為局部變數,day,month,year為成員變數。下面分析一下代碼執行時候的變化:
1. main方法開始執行:int date = 9;
date局部變數,基礎類型,引用和值都存在棧中。
2. Test test = new Test();
test為對象引用,存在棧中,對象(new Test())存在堆中。
3. test.change(date);
i為局部變數,引用和值存在棧中。當方法change執行完成後,i就會從棧中消失。
4. BirthDate d1= new BirthDate(7,7,1970); 
d1 為對象引用,存在棧中,對象(new BirthDate())存在堆中,其中d,m,y為局部變數儲存在棧中,且它們的類型為基礎類型,因此它們的資料也儲存在棧中。 day,month,year為成員變數,它們儲存在堆中(new BirthDate()裡面)。當BirthDate構造方法執行完之後,d,m,y將從棧中消失。
5.main方法執行完之後,date變數,test,d1引用將從棧中消失,new Test(),new BirthDate()將等待記憶體回收。

Java中堆、棧、常量池等概念解析

程式運行時,我們最好對資料儲存到什麼地方做到心中有數。特別要注意的是記憶體的分配。有六個地方都可以儲存資料:
(1) 寄存器。這是最快的儲存地區,因為它位於和其他所有儲存方式不同的地方:處理器內部。然而,寄存器的數量十分有限,所以寄存器是根據需要由編譯器分配。我們對此沒有直接的控制權,也不可能在自己的程式裡找到寄存器存在的任何蹤跡。

(2) 棧(stack)。存放基本類型的變數資料和對象的引用,但對象本身不存放在棧中,而是存放在堆(new 出來的對象)或者常量池中(字串常量對象存放在常量池中。) 駐留於常規RAM(隨機訪問儲存空間)地區,但可通過它的“堆棧指標”獲得處理的直接支援。堆棧指標若向下移,會建立新的記憶體;若向上移,則會釋放那些內 存。這是一種特別快、特別有效資料儲存方式,僅次於寄存器。建立程式時,Java編譯器必須準確地知道堆棧內儲存的所有資料的“長度”以及“存在時 間”。這是由於它必鬚生成相應的代碼,以便向上和向下移動指標。這一限制無疑影響了程式的靈活性,所以儘管有些Java資料要儲存在堆棧裡——特別是對象 控制代碼,但Java對象並不放到其中。

(3) 堆(heap)。存放所有new出來的對象,一種常規用途的記憶體池(也在RAM地區),其中保 存了Java對象。和堆棧不同,“記憶體堆”或“堆”(Heap)最迷人的地方在於編譯器不必知道要從堆裡分配多少儲存空間,也不必知道儲存的資料要在堆 裡停留多長的時間。因此,用堆儲存資料時會得到更大的靈活性。要求建立一個對象時,只需用new命令編製相關的代碼即可。執行這些代碼時,會在堆裡自動進 行資料的儲存。當然,為達到這種靈活性,必然會付出一定的代價:在堆裡分配儲存空間時會花掉更長的時間!

(4) 靜態儲存。存放靜態成員(static定義的),這兒的“靜態”(Static)是指“位於固定位置”(儘管也在RAM裡)。程式運行期間,靜態儲存的資料將隨時等候調用。可用static關鍵字指出一個對象的特定元素是靜態。但Java對象本身永遠都不會置入靜態儲存空間。

(5) 常數儲存。存放字串常量和基本類型常量(public static final)。 常數值通常直接置於程式碼內部。這樣做是安全的,因為它們永遠都不會改變。有的常數需要嚴格地保護,所以可考慮將它們置入唯讀記憶體(ROM)。

(6) 非RAM儲存。若資料完全獨立於一個程式之外,則程式不運行時仍可存在,並在程式的控制範圍之 外。其中兩個最主要的例子便是“流式對象”和“固定對象”。對於流式對象,對象會變成位元組流,通常會發給另一台機器。而對於固定對象,對象儲存在磁碟中。 即使程式中止運行,它們仍可保持自己的狀態不變。對於這些類型的資料存放區,一個特別有用的技巧就是它們能存在於其他媒體中。一旦需要,甚至能將它們恢複成 普通的、基於RAM的對象。Java 1.1提供了對Lightweight persistence的支援。未來的版本甚至可能提供更完整的方案

 

這裡我們主要關心棧,堆和常量池,對於棧和常量池中的對象可以共用,對於堆中的對象不可以共用。棧中的資料大小和生命週期是可以確定的,當沒有引用指向資料時,這個資料就會消失。堆中的對象的由記憶體回收行程負責回收,因此大小和生命週期不需要確定,具有很大的靈活性。
對於字串:其對象的引用都是儲存在棧中的,如果是編譯期已經建立好(直接用雙引號定義的)的就儲存在常量池中,如果是運行期(new出來的)才能確定的就儲存在堆中。對於equals相等的字串,在常量池中永遠只有一份,在堆中有多份。
如以下代碼:

Java代碼

String s1 = "china";  
String s2 = "china";  
String s3 = "china";  
String ss1 = new String("china");  
String ss2 = new String("china");  
String ss3 = new String("china");  
 

這裡解釋一下黃色這3個箭頭,對於通過new產生一個字串(假設為”china”)時,會先去常量池中尋找是否已經有了”china”對象,如果 沒有則在常量池中建立一個此字串對象,然後堆中再建立一個常量池中此”china”對象的拷貝對象。這也就是有道面試題:String s = new String(“xyz”);產生幾個對象?一個或兩個,如果常量池中原來沒有”xyz”,就是兩個。

對於基礎類型的變數和常量:變數和引用儲存在棧中,常量儲存在常量池中。
如以下代碼:

Java代碼

int i1 = 9;  
int i2 = 9;  
int i3 = 9;   
public static final int INT1 = 9;  
public static final int INT2 = 9;  
public static final int INT3 = 9;  

 
對於成員變數和局部變數:成員變數就是方法外部,類的內部定義的變數;局部變數就是方法或語句塊內部定義的變數。局部變數必須初始化。
形式參數是局部變數,局部變數的資料存在於棧記憶體中。棧記憶體中的局部變數隨著方法的消失而消失。
成員變數儲存在堆中的對象裡面,由記憶體回收行程負責回收。
如以下代碼:

Java代碼

class BirthDate {  
    private int day;  
    private int month;  
    private int year;      
    public BirthDate(int d, int m, int y) {  
        day = d;   
        month = m;   
        year = y;  
    }  
    省略get,set方法………  
}  

public class Test{  
    public static void main(String args[]){  
int date = 9;  
        Test test = new Test();        
           test.change(date);   
        BirthDate d1= new BirthDate(7,7,1970);         
    }    

    public void change1(int i){  
        i = 1234;  
    } 


 
對於以上這段代碼,date為局部變數,i,d,m,y都是形參為局部變數,day,month,year為成員變數。下面分析一下代碼執行時候的變化:
1. main方法開始執行:int date = 9;
date局部變數,基礎類型,引用和值都存在棧中。
2. Test test = new Test();
test為對象引用,存在棧中,對象(new Test())存在堆中。
3. test.change(date);
i為局部變數,引用和值存在棧中。當方法change執行完成後,i就會從棧中消失。
4. BirthDate d1= new BirthDate(7,7,1970); 
d1 為對象引用,存在棧中,對象(new BirthDate())存在堆中,其中d,m,y為局部變數儲存在棧中,且它們的類型為基礎類型,因此它們的資料也儲存在棧中。 day,month,year為成員變數,它們儲存在堆中(new BirthDate()裡面)。當BirthDate構造方法執行完之後,d,m,y將從棧中消失。
5.main方法執行完之後,date變數,test,d1引用將從棧中消失,new Test(),new BirthDate()將等待記憶體回收。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.