1.寄存器
最快的儲存區, 由編譯器根據需求進行分配,我們在程式中無法控制。
2. 棧
存放基本類型的變數資料和對象的引用,但對象本身不存放在棧中,而是存放在堆(new 出來的對象)或者常量池中(字串常量對象存放在常量池中。)
3. 堆
存放所有new出來的對象。
4. 靜態域
存放靜態成員(static定義的)
5. 常量池
存放字串常量和基本類型常量(public static final)。
6. 非RAM儲存
硬碟等永久儲存空間
這裡我們主要關心棧,堆和常量池,對於棧和常量池中的對象可以共用,對於堆中的對象不可以共用。棧中的資料大小和生命週期是可以確定的,當沒有引用指向資料時,這個資料就會消失。堆中的對象的由記憶體回收行程負責回收,因此大小和生命週期不需要確定,具有很大的靈活性。
對於字串:其對象的引用都是儲存在棧中的,如果是編譯期已經建立好(直接用雙引號定義的)的就儲存在常量池中,如果是運行期(new出來的)才能確定的就儲存在堆中。對於equals相等的字串,在常量池中永遠只有一份,在堆中有多份。
如以下代碼:
String s1 = "china"; String s2 = "china"; String s3 = "china"; String ss1 = new String("china"); String ss2 = new String("china"); String ss3 = new String("china");
對於基礎類型的變數和常量:變數和引用儲存在棧中,常量儲存在常量池中。
如以下代碼:
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;
對於成員變數和局部變數:成員變數就是方法外部,類的內部定義的變數;局部變數就是方法或語句塊內部定義的變數。局部變數必須初始化。形式參數是局部變數,局部變數的資料存在於棧記憶體中。棧記憶體中的局部變數隨著方法的消失而消失。
成員變數儲存在堆中的對象裡面,由記憶體回收行程負責回收。
如以下代碼:
class BirthDate { privateint day; privateint month; privateint year; public BirthDate(int d, int m, int y) { day = d; month = m; year = y; } 省略get,set方法…… } publicclass Test{ publicstaticvoid main(String args[]){ int date = 9; Test test = new Test(); test.change(date); BirthDate d1= new BirthDate(7,7,1970); } publicvoid change1(int i){ i = 1234;
對於以上這段代碼,date為局部變數,i,d,m,y都是形參為局部變數,day,month,year為成員變數。下面分析一下代碼執行時候的變化:
1. main方法開始執行:
int date = 9;
date局部變數,基礎類型,引用和值都存在棧中。
2. test為對象引用,存在棧中,對象(new Test())存在堆中。
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()將等待記憶體回收。