堆中存什么?棧中存什么裳扯?
堆中存的是對(duì)象抛丽。
棧中存的是基本數(shù)據(jù)類(lèi)型和堆中對(duì)象的引用。
java中堆棧的區(qū)別饰豺?
在Java中亿鲜,棧stack內(nèi)存是用來(lái)存儲(chǔ)函數(shù)的主體和變量名的。Java中的代碼是在函數(shù)體中執(zhí)行的冤吨,每個(gè)函數(shù)主體都會(huì)被放在棧內(nèi)存中蒿柳,比如main函數(shù)。加入main函數(shù)里調(diào)用了其他的函數(shù)漩蟆,比如add()垒探,那么在棧里面的的存儲(chǔ)就是最底層是main,mian上面是add怠李。棧的運(yùn)行時(shí)后入先出的圾叼,所以會(huì)執(zhí)行時(shí)會(huì)先銷(xiāo)毀add,再銷(xiāo)毀main捺癞。
在Java中夷蚊,堆內(nèi)存是用來(lái)存儲(chǔ)實(shí)例的。比如main函數(shù)里面聲明了一個(gè)people的類(lèi)per髓介,people per惕鼓;這個(gè)per是存儲(chǔ)在棧stack內(nèi)存中的,實(shí)例化后(per = new people())唐础;實(shí)例后的對(duì)象實(shí)體是存在堆heap內(nèi)存中的箱歧。棧stack內(nèi)存中存儲(chǔ)的per存儲(chǔ)著指向堆heap內(nèi)存的地址指向矾飞。堆heap內(nèi)存的存在是為了更好的管理內(nèi)存,實(shí)現(xiàn)garbage collection呀邢。當(dāng)per不再指向堆heap內(nèi)存中的實(shí)例的時(shí)候洒沦,garbage collection機(jī)制就會(huì)把這個(gè)堆heap內(nèi)存中的new people()實(shí)例刪除,釋放內(nèi)存
為什么要設(shè)計(jì)堆棧驼鹅?
設(shè)計(jì)原因有兩點(diǎn):⑴棧的存取速度特別快微谓,僅次于寄存器。當(dāng)程序要在內(nèi)存中讀寫(xiě)數(shù)據(jù)時(shí)输钩,先找到棧豺型,再通過(guò)棧的指向到堆內(nèi)存中進(jìn)行數(shù)據(jù)的讀寫(xiě)。由于不用再遍歷速度慢而且數(shù)據(jù)多的堆內(nèi)存买乃,所以程序的運(yùn)行速度會(huì)更快姻氨。
⑵便于內(nèi)存的回收。當(dāng)程序的一些數(shù)據(jù)不用后剪验,就會(huì)丟掉棧內(nèi)存中相應(yīng)的數(shù)據(jù)肴焊,此時(shí)以前指向的堆內(nèi)存空間就沒(méi)有了棧的指向,變成了垃圾功戚。這時(shí)Java虛擬機(jī)的垃圾回收機(jī)制就會(huì)將這塊沒(méi)有棧指向的堆內(nèi)存空間回收娶眷,供給以后的程序使用。
為什么要分為堆和棧啸臀?
第一届宠,從軟件設(shè)計(jì)的角度看,棧代表了處理邏輯乘粒,而堆代表了數(shù)據(jù)豌注。這樣分開(kāi),使得處理邏輯更為清晰灯萍。分而治之的思想轧铁。這種隔離、模塊化的思想在軟件設(shè)計(jì)的方方面面都有體現(xiàn)旦棉。
第二齿风,堆與棧的分離,使得堆中的內(nèi)容可以被多個(gè)棧共享(也可以理解為多個(gè)線程訪問(wèn)同一個(gè)對(duì)象)绑洛。這種共享的收益是很多的聂宾。一方面這種共享提供了一種有效的數(shù)據(jù)交互方式(如:共享內(nèi)存),另一方面诊笤,堆中的共享常量和緩存可以被所有棧訪問(wèn),節(jié)省了空間巾陕。
第三讨跟,棧因?yàn)檫\(yùn)行時(shí)的需要纪他,比如保存系統(tǒng)運(yùn)行的上下文,需要進(jìn)行地址段的劃分晾匠。由于棧只能向上增長(zhǎng)茶袒,因此就會(huì)限制住棧存儲(chǔ)內(nèi)容的能力。而堆不同凉馆,堆中的對(duì)象是可以根據(jù)需要?jiǎng)討B(tài)增長(zhǎng)的薪寓,因此棧和堆的拆分,使得動(dòng)態(tài)增長(zhǎng)成為可能澜共,相應(yīng)棧中只需記錄堆中的一個(gè)地址即可向叉。
第四,面向?qū)ο缶褪嵌押蜅5耐昝澜Y(jié)合嗦董。
更多理解可以參考這篇文章:
http://blog.jobbole.com/75321/