? ? 在J2SE中,通過(guò)進(jìn)行內(nèi)存分析糟趾,可以讓我們更好的理解程序在內(nèi)存中的內(nèi)存分配問(wèn)題酝润,也能讓我們更好的理解我們的代碼是怎么運(yùn)行的。
? ? 對(duì)于我個(gè)人來(lái)說(shuō)分析內(nèi)存也是一件很有趣的事情址芯。所以下面通過(guò)一個(gè)例子來(lái)總結(jié)一下程序運(yùn)行過(guò)程中的內(nèi)存分配。
程序?qū)嵗?/a>
public class Point{
? ? double x,y,z;?
? ? Point(double_x,double_y,double_z)? ? {? ? ? ?
? ? ? ? x=_x;? ? ? ?
? ? ? ? y=_y;? ? ? ?
? ? ? ? z=_z;?
? ? }
? ? void setX(double_x){? ? ? ?
? ? ? ? x=_x;?
? ? }
? ? double getDistance(Point p){
? ? ? ? return(x-p.x)*(x-p.x)+(y-p.y)*(y-p.y)+(z-p.z)*(z-p.z);
? ? }
}
public class TestPoint{
? ? public static void main(String[] args){? ? ? ?
? ? ? ? Point p=newPoint(1.0,2.0,3.0);? ? ? ? //第一步
? ? ? ? Point p1=newPoint(0.0,0.0,0.0);? ? ? ? //第二步
? ? ? ? System.out.println(p.getDistance(p1));? ? ? ? //第三步
? ? ? ? p.setX(5.0);? ? ? ? //第四步
? ? ? ? System.out.println(p.getDistance(newPoint(1.0,1.0,1.0)));? ? ? ? //第五步
? ? }
}
內(nèi)存分析
第一步
? ? 首先定義了一個(gè)變量,所以在棧內(nèi)存中會(huì)分配一塊空間來(lái)存儲(chǔ)變量,變量名為P,它指向了一個(gè)new出來(lái)的對(duì)象窜觉,這個(gè)new出來(lái)的對(duì)象谷炸,當(dāng)然是放到堆內(nèi)存。
? ? 在執(zhí)行的過(guò)程中調(diào)用了Point的構(gòu)造方法禀挫,在執(zhí)行構(gòu)造方法的過(guò)程中,在棧內(nèi)存中臨時(shí)分配了三個(gè)空間分別存儲(chǔ)_x,_y,_z,然后再把這三個(gè)值賦值給點(diǎn)對(duì)象中的x,y,z.
? ? 賦值完成后這些臨時(shí)分配的變量將會(huì)被取消.所以第一步執(zhí)行完后的內(nèi)存分配如下圖:
第二步
? ? 第二步的執(zhí)行過(guò)程和第一步類似,不再做贅述,該步執(zhí)行完后的內(nèi)存分配如下圖:
第三步
? ? 該步調(diào)用了一個(gè)方法,從方法中可以看到有一個(gè)形參旬陡,所以首先在棧內(nèi)存中臨時(shí)分配一塊空間,保存這個(gè)形參,也就是局部變量P.再看getDistance()方法中我們實(shí)際傳入的是P1语婴,所以將p引用P1引用的對(duì)象(這里注意區(qū)分此刻分配的局部變量P和第一步new出來(lái)的P不是一回事)描孟,該步的方法是有返回值的,所以需要在棧中分配空間來(lái)保存這個(gè)返回值砰左。內(nèi)存分配圖如下:
? ? 將返回值打印后,臨時(shí)分配的內(nèi)存空間取消画拾,其內(nèi)存分配圖和第二步執(zhí)行完后的內(nèi)存分配圖是相同的。
第四步
? ? 從調(diào)用的方法上看,首先要在棧中分配一個(gè)臨時(shí)變量_x,值為5.0.然后再將值賦值給P指向的對(duì)象菜职,執(zhí)行完后臨時(shí)變量被取消。內(nèi)存分配圖如下:
第五步
? ? 此表達(dá)式比較復(fù)雜,當(dāng)一個(gè)表達(dá)式復(fù)雜的時(shí)候,我們要從里向外分析旗闽。
? ? 基于此原則,分析得到,首先要在堆內(nèi)存中new一個(gè)對(duì)象,這時(shí)這個(gè)對(duì)象并沒(méi)有被引用酬核。
? ? 然后執(zhí)行方法蜜另,上面我們已經(jīng)分析過(guò)了,執(zhí)行這個(gè)方法我們需要先在棧中臨時(shí)分配一個(gè)變量P,此時(shí)需要將new出的對(duì)象給臨時(shí)變量P,也就是說(shuō)這個(gè)P引用我們剛剛new出的那個(gè)對(duì)象嫡意。
? ? 接著計(jì)算兩點(diǎn)的距離,這時(shí)需要一個(gè)臨時(shí)變量來(lái)保存返回來(lái)的數(shù)據(jù)举瑰。
? ? 方法執(zhí)行完后,所有臨時(shí)分配的內(nèi)存被取消.這時(shí)要注意的是我們new出來(lái)的對(duì)象是通過(guò)垃圾回收機(jī)制取消的。
? ? 內(nèi)存分析如下圖:
程序執(zhí)行完后的內(nèi)存分配
? ? 最后展示一張程序執(zhí)行完后的內(nèi)存分配圖:
總結(jié)
? ? 對(duì)于內(nèi)存分析之前個(gè)人認(rèn)為較為抽象,但是經(jīng)過(guò)仔細(xì)分析后發(fā)現(xiàn)內(nèi)存分析其實(shí)并不復(fù)雜,只要能夠看明白以下這幾點(diǎn),對(duì)于一些簡(jiǎn)單的內(nèi)存分析基本上可以說(shuō)已經(jīng)沒(méi)有問(wèn)題了蔬螟。
? ? (1)形參要臨時(shí)分配空間,程序執(zhí)行完后要 取消空間分配此迅。
? ? (2)對(duì)于復(fù)雜的程序要從里到外分析,一步步來(lái)旧巾。
? ? (3)有返回值的方法,其返回值也是要分配給一個(gè)臨時(shí)變量的耸序,同樣的使用完后要取消。
? ? (4)臨時(shí)new出來(lái)的對(duì)象是利用垃圾回收機(jī)制取消的鲁猩。
? ? (5)靜態(tài)成員變量時(shí)放在哪里的,是怎么分配內(nèi)存的(這個(gè)例子并沒(méi)有展示靜態(tài)成員變量的分配)坎怪。