Java內(nèi)存區(qū)域分為:
- 方法區(qū)(Method Area):所有線程共有
- 虛擬機(jī)棧(VM Stack):每一個(gè)線程獨(dú)有
- 本地方法棧(Native Method Stack):HotSpot虛擬機(jī)直接將本地方法棧和虛擬機(jī)棧合二為一。
- 堆(heap):所有線程共有
- 程序計(jì)數(shù)器(Program Counter Register):每個(gè)線程獨(dú)有壳繁,只記錄非native方法
/**
* 運(yùn)行時(shí), jvm 把AppMain的信息都放入方法區(qū)
*/
public class AppMain {
/**
* main 方法放入棧區(qū)震捣。
*/
public static void main(String[] args) {
Sample test1 ;
System.out.println("Sample test1 statement");
test1= new Sample(" 測(cè)試1 "); //test1是引用荔棉,所以放到棧區(qū)里, Sample是自定義對(duì)象應(yīng)該放到堆里面
Sample test2 = new Sample(" 測(cè)試2 ");
test1.printName();
test2.printName();
}
}
/**
* 運(yùn)行時(shí), jvm 把Sample的信息都放入方法區(qū)
*/
class Sample {
static{
System.out.println("Sample class loaded");
}
//new Sample實(shí)例后蒿赢, name 引用放入堆里润樱, name 對(duì)象放入堆里
private String name;
public Sample(String name) {
this.name = name;
}
/**
* print方法本身放入 方法區(qū)里。
*/
public void printName() {
System.out.println(name);
}
}
程序運(yùn)行過程:
Jvm首先加載AppMain類進(jìn)方法區(qū)羡棵,然后執(zhí)行main()方法壹若。test1是局部變量會(huì)加載到棧中,然后實(shí)例化Sample對(duì)象晾腔,此時(shí)需要加載Sample類到方法區(qū)舌稀,然后在堆中new出一個(gè)Sample的對(duì)象,地址交給test1在棧中保存灼擂。**然后調(diào)用printName()方法,Jvm將繼續(xù)執(zhí)行后續(xù)指令壁查,在堆區(qū)里繼續(xù)創(chuàng)建另一個(gè)Sample實(shí)例,然后依次執(zhí)行它們的printName()方法剔应。當(dāng)JAVA虛擬機(jī)執(zhí)行test1.printName()方法時(shí)睡腿,JAVA虛擬機(jī)根據(jù)局部變量test1持有的引用,定位到堆區(qū)中的Sample實(shí)例峻贮,再根據(jù)Sample實(shí)例持有的引用反粥,定位到方法去中Sample類的類型信息漓藕,從而獲得printName()方法的字節(jié)碼护侮,接著執(zhí)行printName()方法包含的指令.
(printName()會(huì)進(jìn)去棧中嗎活烙?在棧中怎么存放?Sample的name屬性是放在堆中還是棧中船万?)
方法進(jìn)入棧刻撒,形成一個(gè)棧幀,name屬性放在堆中耿导,只有方法中的變量進(jìn)入棧中声怔。
疑問:到底是test1變量在聲明時(shí)加載Sample還是在new的時(shí)候加載Sample到方法區(qū)?
經(jīng)過打印log發(fā)現(xiàn)Sample是在new的時(shí)候加載,在聲明的時(shí)候并不加載舱呻。
深入理解JVM(1) : Java內(nèi)存區(qū)域劃分
Java里的堆(heap)棧(stack)和方法區(qū)(method)
http://www.cnblogs.com/gw811/archive/2012/10/18/2730117.html
http://www.cnblogs.com/vamei/archive/2013/04/28/3048353.html