Android之内存机制分析-Android堆和栈
1、dalvik的Heap和Stack
這里說的只是dalvik ?java部分的內(nèi)存,實(shí)際上除了dalvik部分,還有native。這個(gè)以后再說。
下面針對(duì)上面列出的數(shù)據(jù)類型進(jìn)行說明,只有了解了我們申請(qǐng)的數(shù)據(jù)在哪里,才能更好掌控我們自己的程序。
?
2、對(duì)象實(shí)例數(shù)據(jù)
實(shí)際上是保存對(duì)象實(shí)例的屬性,屬性的類型和對(duì)象本身的類型標(biāo)記等,但是不保存實(shí)例的方法。實(shí)例的方法是屬于數(shù)據(jù)指令,是保存在Stack里面,也就是上面表格里面的類方法。
對(duì)象實(shí)例在Heap中分配好以后,會(huì)在stack中保存一個(gè)4字節(jié)的Heap內(nèi)存地址,用來查找對(duì)象的實(shí)例。因?yàn)樵赟tack里面會(huì)用到Heap的實(shí)例,特別是調(diào)用實(shí)例的時(shí)候需要傳入一個(gè)this指針。
?
3、方法內(nèi)部變量
類方法的內(nèi)部變量分為兩種情況:簡(jiǎn)單類型保存在Stack中;對(duì)象類型在Stack中保存地址,在Heap 中保存值。
?
4、非靜態(tài)方法和靜態(tài)方法
非靜態(tài)方法有一個(gè)隱含的傳入?yún)?shù),這個(gè)參數(shù)是dalvik虛擬機(jī)傳進(jìn)去的,這個(gè)隱含參數(shù)就是對(duì)象實(shí)例在Stack中的地址指針。因此非靜態(tài)方法(在Stack中的指令代碼)總是可以找到自己的專用數(shù)據(jù)(在Heap 中的對(duì)象屬性值)。當(dāng)然非靜態(tài)方法也必須獲得該隱含參數(shù),因此非靜態(tài)方法在調(diào)用前,必須先new一個(gè)對(duì)象實(shí)例,獲得Stack中的地址指針,否則dalvik虛擬機(jī)將無法將隱含參數(shù)傳給非靜態(tài)方法。
靜態(tài)方法沒有隱含參數(shù),因此也不需要new對(duì)象,只要class文件被ClassLoader load進(jìn)入JVM的Stack,該靜態(tài)方法即可被調(diào)用。所以我們可以直接使用類名調(diào)用類的方法。當(dāng)然此時(shí)靜態(tài)方法是存取不到Heap 中的對(duì)象屬性的。
?
5、靜態(tài)屬性和動(dòng)態(tài)屬性
靜態(tài)屬性是保存在Stack中的,而不同于動(dòng)態(tài)屬性保存在Heap 中。正因?yàn)槎际窃赟tack中,而Stack中指令和數(shù)據(jù)都是定長(zhǎng)的,因此很容易算出偏移量,所以類方法(靜態(tài)和非靜態(tài))都可以訪問到類的靜態(tài)屬性。也正因?yàn)殪o態(tài)屬性被保存在Stack中,所以具有了全局屬性。
?
6、總結(jié)
Java 的堆是一個(gè)運(yùn)行時(shí)數(shù)據(jù)區(qū),類的(對(duì)象從中分配空間。這些對(duì)象通過new、newarray、anewarray和multianewarray等指令建立,它們不需要程序代碼來顯式的釋放。堆是由垃圾回收來負(fù)責(zé)的,堆的優(yōu)勢(shì)是可以動(dòng)態(tài)地分配內(nèi)存大小,生存期也不必事先告訴編譯器,因?yàn)樗窃谶\(yùn)行時(shí)動(dòng)態(tài)分配內(nèi)存的,Java的垃圾收集器會(huì)自動(dòng)收走這些不再使用的數(shù)據(jù)。但缺點(diǎn)是,由于要在運(yùn)行時(shí)動(dòng)態(tài)分配內(nèi)存,存取速度較慢。" ??????? “棧的優(yōu)勢(shì)是,存取速度比堆要快,僅次于寄存器,棧數(shù)據(jù)可以共享。但缺點(diǎn)是,存在棧中的數(shù)據(jù)大小與生存期必須是確定的,缺乏靈活性。棧中主要存放一些基本類型的變量(,int, short, long, byte, float, double, boolean, char)和對(duì)象句柄。 對(duì)比上面的解析可以看出,其實(shí)Java處理Heap和Stack的大致原理跟C++是一樣的。只是多了一個(gè)內(nèi)存回收機(jī)制,讓程序員不用主動(dòng)調(diào)用delete釋放內(nèi)存。就像在C++里面,一般使用new申請(qǐng)的內(nèi)存才會(huì)放到堆里面,而一般的臨時(shí)變量都是放到棧里面去。總結(jié)
以上是生活随笔為你收集整理的Android之内存机制分析-Android堆和栈的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Android之线程池
- 下一篇: Android之Launcher分析和修