北大青鳥(niǎo)通州校區(qū)學(xué)術(shù)部提供:
java中內(nèi)存的分配方式有兩種,一種是在堆中分配,一種是在堆棧中分配,所有new出來(lái)的對(duì)象都是在堆中分配的,函數(shù)中參數(shù)的傳遞是在棧中分配的。通常情況下堆的內(nèi)存可以很大,比如32位操作系統(tǒng)中的虛擬內(nèi)存都可以被堆所使用(當(dāng)內(nèi)存緊張的時(shí)候甚至硬盤(pán)都可以是堆的存儲(chǔ)空間),而堆棧的內(nèi)存分配是有限的。這和c++中內(nèi)存分配差不多(c++中還要有另一種方式用于全局變量或者局部靜態(tài)變量的內(nèi)存分配,這里就不說(shuō)了)。java中有幾種基本類型如int,float,double,char,byte等,他們不是對(duì)象,除此之外一切都是對(duì)象,所有的對(duì)象都是在堆上分配的。java中對(duì)象數(shù)組是什么,和c++類似,是句柄數(shù)組或者叫指針數(shù)組,里面保存的是每個(gè)元素的地址。和c++中不同,java沒(méi)有操作符重載和拷貝構(gòu)造函數(shù)(如果不了解這些也沒(méi)有關(guān)系),因此當(dāng)創(chuàng)建對(duì)象或者對(duì)已經(jīng)創(chuàng)建的對(duì)象賦值時(shí)(注意是對(duì)象,不是基本類型):Object a=new Object 和Object a=b(b是Object的子類型或者同類型)時(shí),進(jìn)行的是對(duì)象地址的傳遞并復(fù)制。這就是所說(shuō)的句柄的傳遞和賦值。句柄里存儲(chǔ)的就是對(duì)象的地址,句柄就是指針,只不過(guò)是你無(wú)法得到的地址,java就是通過(guò)這一點(diǎn)巧妙的將指針隱藏起來(lái)。當(dāng)對(duì)象作為參數(shù)傳遞到方法中時(shí),傳遞的就是對(duì)象的地址,而行參中保存的是實(shí)參地址的副本(這就是最關(guān)鍵的地方,也是值傳遞,值傳遞就是將實(shí)參的值的副本作為行參)如:
public class Example{
int i=0;
}
public class A{
public int i=0;
public Example add0(Example e)
{
e.i++;
return e;
}
public void add1(Example e)
{
e.i++;
}
public void modify0(Example e)
{
Example b=e;//將e行參對(duì)象的地址賦給句柄b
b.i++;//也同時(shí)修改了e.i和實(shí)參的值
}
public void modify1(Example e)
{
e=new Example();
e.i++;
}
public static void main(String[] args)
{
Example ex=new Example();
A a=new A();
a=a.add0(ex);//等價(jià)于a.add0(ex),無(wú)需返回值,因?yàn)橥ㄟ^(guò)傳遞的對(duì)象地址(句柄),直接修改了ex中i的值
a.add1(ex);//add0,add1都在其中的方法體中直接修改了ex.i的值,因此add0的返回值有點(diǎn)多余
a.modify0(ex);//對(duì)ex所產(chǎn)生的影響同add1
a.modify1(ex);//對(duì)ex沒(méi)有產(chǎn)生任何影響(而且這就是等價(jià)于什么也沒(méi)有做).
這可能會(huì)讓一部分人搞不清了。為什么呢?因?yàn)槭菍?duì)象地址的副本值傳遞,在modify1中e=new Example();實(shí)際上e僅僅是保存ex對(duì)象地址的副本的一個(gè)句柄,當(dāng)對(duì)e賦值時(shí)僅僅是對(duì)堆棧中e的賦值(對(duì)ex指針副本的變量e賦值),而并沒(méi)有改變ex的句柄的指向,當(dāng)方法調(diào)用完畢堆棧彈出,e就將要被垃圾回收,沒(méi)有任何用處。當(dāng)然你可以將它作為返回值,這就是另外一回事了。
}
}
這里比較繞,如果你能明白這個(gè)原理,那么你就可以寫(xiě)出合理并且高效的程序,并且可以避免一些潛在的邏輯錯(cuò)誤,如:對(duì)象在方法中被改動(dòng)了,可能你還不知道!記住c++在這一點(diǎn)上和java有很大的不同,c++默認(rèn)的是值傳遞,行參會(huì)按照位復(fù)制實(shí)參(如果用指針或者引用就和java很類似了),在方法中作為參數(shù)傳遞對(duì)象,java更象是c++中傳遞引用,當(dāng)然還是有區(qū)別的,那就是c++中對(duì)象的引用不可再賦值為另一個(gè)對(duì)象,也就是說(shuō)modify1中的再賦值對(duì)引用是不可以的。如果你對(duì)c++不了解,那么就當(dāng)我什么也沒(méi)有說(shuō),和c++的比較只是為了幫助更好的理解(針對(duì)熟悉c++而不熟悉java的人)。我本人也對(duì)c++了解甚少,平時(shí)主要工作側(cè)重于java。因此如果哪位高人發(fā)現(xiàn)以上解釋有什么錯(cuò)誤請(qǐng)不吝賜教。
引申到克隆技術(shù)java中的所有對(duì)象都是Object類的子類,Object類定義了protected clone()方法,它的作用和c++中按位復(fù)制是一樣的,因此同樣會(huì)帶來(lái)如果對(duì)象中包含另一個(gè)對(duì)象(注意是對(duì)象不是基本數(shù)據(jù)類型,基本數(shù)據(jù)類型直接就會(huì)被復(fù)制)的指針(java中的句柄),clone并沒(méi)有將被包含的對(duì)象clone,而是復(fù)制了被包含對(duì)象的句柄或者說(shuō)指針。因此并不能認(rèn)為復(fù)制出來(lái)的對(duì)象就可以隨心所欲的修改,因?yàn)樗捅籧lone的對(duì)象都包含同一個(gè)對(duì)象,因此可能會(huì)引起潛在的沖突問(wèn)題。至于深度clone的方法很簡(jiǎn)單,就是在子類中覆蓋父類Object類中clone方法,保證每一個(gè)被包含的對(duì)象都被按照位被clone。如果包含的數(shù)據(jù)全部是基本類型數(shù)據(jù),那么就什么也不用做了。深度clone還有另一種方法就是利用Serializable,但是對(duì)象中被transient關(guān)鍵字修飾的變量是不會(huì)被序列化的. 因?yàn)閏lone用到的地方并不多,就不多說(shuō)了。但是當(dāng)你遇到的時(shí)候,一定要小心。