真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

如何理解JVM中的Stack和Heap

如何理解JVM中的Stack和Heap,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

成都創(chuàng)新互聯(lián)公司長期為上千多家客戶提供的網(wǎng)站建設(shè)服務(wù),團隊從業(yè)經(jīng)驗10年,關(guān)注不同地域、不同群體,并針對不同對象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為西青企業(yè)提供專業(yè)的成都做網(wǎng)站、網(wǎng)站設(shè)計,西青網(wǎng)站改版等技術(shù)服務(wù)。擁有十載豐富建站經(jīng)驗和眾多成功案例,為您定制開發(fā)。

在JVM中,內(nèi)存分為兩個部分,Stack(棧)和Heap(堆),這里,我們從JVM的內(nèi)存管理原理的角度來認識Stack和Heap,并通過這些原理認清Java中靜態(tài)方法和靜態(tài)屬性的問題。

一般,JVM的內(nèi)存分為兩部分:Stack和Heap。

Stack(棧)是JVM的內(nèi)存指令區(qū)。Stack管理很簡單,push一定長度字節(jié)的數(shù)據(jù)或者指令,Stack指針壓棧相應(yīng)的字節(jié)位移;pop一定字節(jié)長度數(shù)據(jù)或者指令,Stack指針彈棧。Stack的速度很快,管理很簡單,并且每次操作的數(shù)據(jù)或者指令字節(jié)長度是已知的。所以Java 基本數(shù)據(jù)類型,Java 指令代碼,常量都保存在Stack中。

Heap(堆)是JVM的內(nèi)存數(shù)據(jù)區(qū)。Heap 的管理很復(fù)雜,每次分配不定長的內(nèi)存空間,專門用來保存對象的實例。在Heap 中分配一定的內(nèi)存來保存對象實例,實際上也只是保存對象實例的屬性值,屬性的類型和對象本身的類型標記等,并不保存對象的方法(方法是指令,保存在Stack中),在Heap 中分配一定的內(nèi)存保存對象實例和對象的序列化比較類似。而對象實例在Heap 中分配好以后,需要在Stack中保存一個4字節(jié)的Heap 內(nèi)存地址,用來定位該對象實例在Heap 中的位置,便于找到該對象實例。

由于Stack的內(nèi)存管理是順序分配的,而且定長,不存在內(nèi)存回收問題;而Heap 則是隨機分配內(nèi)存,不定長度,存在內(nèi)存分配和回收的問題;因此在JVM中另有一個GC進程,定期掃描Heap ,它根據(jù)Stack中保存的4字節(jié)對象地址掃描Heap ,定位Heap 中這些對象,進行一些優(yōu)化(例如合并空閑內(nèi)存塊什么的),并且假設(shè)Heap 中沒有掃描到的區(qū)域都是空閑的,統(tǒng)統(tǒng)refresh(實際上是把Stack中丟失了對象地址的無用對象清除了),這就是垃圾收集的過程;關(guān)于垃圾收集的更深入講解請參考51CTO之前的文章《JVM內(nèi)存模型及垃圾收集策略解析》。

如何理解JVM中的Stack和Heap 
JVM的體系結(jié)構(gòu)

我們首先要搞清楚的是什么是數(shù)據(jù)以及什么是指令。然后要搞清楚對象的方法和對象的屬性分別保存在哪里。

1)方法本身是指令的操作碼部分,保存在Stack中;

2)方法內(nèi)部變量作為指令的操作數(shù)部分,跟在指令的操作碼之后,保存在Stack中(實際上是簡單類型保存在Stack中,對象類型在Stack中保存地址,在Heap 中保存值);上述的指令操作碼和指令操作數(shù)構(gòu)成了完整的Java 指令。

3)對象實例包括其屬性值作為數(shù)據(jù),保存在數(shù)據(jù)區(qū)Heap 中。

非靜態(tài)的對象屬性作為對象實例的一部分保存在Heap 中,而對象實例必須通過Stack中保存的地址指針才能訪問到。因此能否訪問到對象實例以及它的非靜態(tài)屬性值完全取決于能否獲得對象實例在Stack中的地址指針。

非靜態(tài)方法和靜態(tài)方法的區(qū)別:

非靜態(tài)方法有一個和靜態(tài)方法很重大的不同:非靜態(tài)方法有一個隱含的傳入?yún)?shù),該參數(shù)是JVM給它的,和我們怎么寫代碼無關(guān),這個隱含的參數(shù)就是對象實例在Stack中的地址指針。因此非靜態(tài)方法(在Stack中的指令代碼)總是可以找到自己的專用數(shù)據(jù)(在Heap 中的對象屬性值)。當然非靜態(tài)方法也必須獲得該隱含參數(shù),因此非靜態(tài)方法在調(diào)用前,必須先new一個對象實例,獲得Stack中的地址指針,否則JVM將無法將隱含參數(shù)傳給非靜態(tài)方法。

靜態(tài)方法無此隱含參數(shù),因此也不需要new對象,只要class文件被ClassLoader load進入JVM的Stack,該靜態(tài)方法即可被調(diào)用。當然此時靜態(tài)方法是存取不到Heap 中的對象屬性的。

總結(jié)一下該過程:當一個class文件被ClassLoader load進入JVM后,方法指令保存在Stack中,此時Heap 區(qū)沒有數(shù)據(jù)。然后程序技術(shù)器開始執(zhí)行指令,如果是靜態(tài)方法,直接依次執(zhí)行指令代碼,當然此時指令代碼是不能訪問Heap 數(shù)據(jù)區(qū)的;如果是非靜態(tài)方法,由于隱含參數(shù)沒有值,會報錯。因此在非靜態(tài)方法執(zhí)行前,要先new對象,在Heap 中分配數(shù)據(jù),并把Stack中的地址指針交給非靜態(tài)方法,這樣程序技術(shù)器依次執(zhí)行指令,而指令代碼此時能夠訪問到Heap 數(shù)據(jù)區(qū)了。

靜態(tài)屬性和動態(tài)屬性

前面提到對象實例以及動態(tài)屬性都是保存在Heap 中的,而Heap 必須通過Stack中的地址指針才能夠被指令(類的方法)訪問到。因此可以推斷出:靜態(tài)屬性是保存在Stack中的,而不同于動態(tài)屬性保存在Heap 中。正因為都是在Stack中,而Stack中指令和數(shù)據(jù)都是定長的,因此很容易算出偏移量,也因此不管什么指令(類的方法),都可以訪問到類的靜態(tài)屬性。也正因為靜態(tài)屬性被保存在Stack中,所以具有了全局屬性。

在JVM中,靜態(tài)屬性保存在Stack指令內(nèi)存區(qū),動態(tài)屬性保存在Heap數(shù)據(jù)內(nèi)存區(qū)。

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對創(chuàng)新互聯(lián)的支持。


新聞標題:如何理解JVM中的Stack和Heap
文章URL:http://weahome.cn/article/geggcj.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部