新聞中心
昨天用Gallery做了一個(gè)圖片瀏覽選擇開(kāi)機(jī)畫(huà)面的功能,當(dāng)我加載的圖片多了就出現(xiàn)OOM問(wèn)題。以前也出現(xiàn)過(guò)這個(gè)問(wèn)題,那時(shí)候并沒(méi)有深究。這次打算好好分析一下Android的內(nèi)存機(jī)制。

我們提供的服務(wù)有:成都網(wǎng)站制作、網(wǎng)站建設(shè)、外貿(mào)網(wǎng)站建設(shè)、微信公眾號(hào)開(kāi)發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、稱多ssl等。為上1000+企事業(yè)單位解決了網(wǎng)站和推廣的問(wèn)題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的稱多網(wǎng)站制作公司
因?yàn)槲乙郧笆亲鯲C++開(kāi)發(fā),因此對(duì)C++在Window下的內(nèi)存機(jī)制還是比較了解。不過(guò)轉(zhuǎn)到Android后,一直都沒(méi)有刻意去處理內(nèi)存問(wèn)題,因?yàn)槟X子里一直想著Java的GC機(jī)制。不過(guò)現(xiàn)在想想,自己對(duì)Android的GC和內(nèi)存管理并不了解,自己寫(xiě)的代碼在內(nèi)存哪里運(yùn)行都不清楚,心里不淡定啊。。。。
畢竟我以前寫(xiě)C++的時(shí)候,什么時(shí)候在哪里申請(qǐng)內(nèi)存,什么時(shí)候釋放內(nèi)存,會(huì)不會(huì)棧溢出或者堆內(nèi)存泄露都了如指掌。言歸正傳,今天打算先了解一下Android的堆和棧跟C++有何區(qū)別。
1、dalvik的Heap和Stack
這里說(shuō)的只是dalvik java部分的內(nèi)存,實(shí)際上除了dalvik部分,還有native。這個(gè)以后再說(shuō)。
下面針對(duì)上面列出的數(shù)據(jù)類(lèi)型進(jìn)行說(shuō)明,只有了解了我們申請(qǐng)的數(shù)據(jù)在哪里,才能更好掌控我們自己的程序。
2、對(duì)象實(shí)例數(shù)據(jù)
實(shí)際上是保存對(duì)象實(shí)例的屬性,屬性的類(lèi)型和對(duì)象本身的類(lèi)型標(biāo)記等,但是不保存實(shí)例的方法。實(shí)例的方法是屬于數(shù)據(jù)指令,是保存在Stack里面,也就是上面表格里面的類(lèi)方法。
對(duì)象實(shí)例在Heap中分配好以后,會(huì)在stack中保存一個(gè)4字節(jié)的Heap內(nèi)存地址,用來(lái)查找對(duì)象的實(shí)例。因?yàn)樵赟tack里面會(huì)用到Heap的實(shí)例,特別是調(diào)用實(shí)例的時(shí)候需要傳入一個(gè)this指針。
3、方法內(nèi)部變量
類(lèi)方法的內(nèi)部變量分為兩種情況:簡(jiǎn)單類(lèi)型保存在Stack中;對(duì)象類(lèi)型在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ī)將無(wú)法將隱含參數(shù)傳給非靜態(tài)方法。
靜態(tài)方法沒(méi)有隱含參數(shù),因此也不需要new對(duì)象,只要class文件被ClassLoader load進(jìn)入JVM的Stack,該靜態(tài)方法即可被調(diào)用。所以我們可以直接使用類(lèi)名調(diào)用類(lèi)的方法。當(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)的,因此很容易算出偏移量,所以類(lèi)方法(靜態(tài)和非靜態(tài))都可以訪問(wèn)到類(lèi)的靜態(tài)屬性。也正因?yàn)殪o態(tài)屬性被保存在Stack中,所以具有了全局屬性。
6、總結(jié)
Java的堆是一個(gè)運(yùn)行時(shí)數(shù)據(jù)區(qū),類(lèi)的(對(duì)象從中分配空間。這些對(duì)象通過(guò)new、newarray、anewarray和multianewarray等指令建立,它們不需要程序代碼來(lái)顯式的釋放。堆是由垃圾回收來(lái)負(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ù)大小與生存期必須是確定的,缺乏靈活性。棧中主要存放一些基本類(lèi)型的變量(,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í)變量都是放到棧里面去。
今天主要是說(shuō)說(shuō)Android的dalvik里面的堆和棧的區(qū)別,以及存放哪些數(shù)據(jù)。粗了dalvik內(nèi)存外, Android還有個(gè)native內(nèi)存的概念。這個(gè)下次會(huì)繼續(xù)講解。我剛開(kāi)始分析Android的內(nèi)存機(jī)制,如果閱讀過(guò)程中發(fā)現(xiàn)任何問(wèn)題請(qǐng)留言指出,謝謝!
文章標(biāo)題:Android內(nèi)存機(jī)制分析上篇:了解Android堆和棧
分享網(wǎng)址:http://m.fisionsoft.com.cn/article/dpcpigh.html


咨詢
建站咨詢
