當(dāng)前位置:首頁(yè) > 嵌入式培訓(xùn) > 嵌入式學(xué)習(xí) > 講師博文 > Android教學(xué)中遇到的內(nèi)存泄露和內(nèi)存溢出
在Android中每個(gè)App的內(nèi)存是有限的,一般來(lái)講,在開(kāi)發(fā)的過(guò)程中我們要避免無(wú)謂的內(nèi)存占用。實(shí)際教學(xué)過(guò)程中,有一個(gè)很簡(jiǎn)單但是很多學(xué)生經(jīng)常會(huì)犯的錯(cuò)誤,在此指出。
觀察以下代碼:
學(xué)生乍一看,沒(méi)有太大的問(wèn)題。當(dāng)然,這個(gè)也反應(yīng)出了很多學(xué)生的一個(gè)不好的習(xí)慣。
首先,Drawable雖然是一種抽象的圖像概念,但是它確實(shí)是占用不少資源的,使用static關(guān)鍵字來(lái)修飾,雖然在編寫(xiě)的過(guò)程中比較省事,但是導(dǎo)致的問(wèn)題是static修飾的sBackground資源不會(huì)輕易釋放,更為嚴(yán)重的是,這個(gè)sBackground的Drawable持有對(duì)其Activity的引用。
這樣就導(dǎo)致這個(gè)Activity的實(shí)例也不會(huì)被釋放,導(dǎo)致內(nèi)存泄露。
這個(gè)問(wèn)題非常嚴(yán)重,但是很多學(xué)生現(xiàn)在的階段并不是很關(guān)注,因此static關(guān)鍵字的教學(xué)的過(guò)程中應(yīng)該進(jìn)行再反思。
關(guān)于上文的內(nèi)存泄露,特別是內(nèi)存溢出,兩者之間的關(guān)系來(lái)說(shuō)明一下。
實(shí)際上來(lái)說(shuō),泄露和溢出這兩個(gè)形容并不是很好區(qū)分。首先這兩前者是一個(gè)狀態(tài),后者是一個(gè)結(jié)果。
內(nèi)存泄露是應(yīng)用使用資源后沒(méi)有及時(shí)釋放(如本文中例子),導(dǎo)致內(nèi)存中一直持有多余的內(nèi)存占用,這個(gè)狀態(tài)的累積會(huì)導(dǎo)致可用內(nèi)存越來(lái)越少。舉例:一個(gè)人的資產(chǎn)有一100萬(wàn),但是他想買(mǎi)一輛50萬(wàn)的車(chē),雖然有一百萬(wàn)的資產(chǎn),但是身上帶著的銀行卡里只有100元,跟老板口舌一下,也能買(mǎi),但是這個(gè)債務(wù)會(huì)一直存在。如果一直這樣消費(fèi),債務(wù)會(huì)滾雪球,早晚有崩潰的一天。
內(nèi)存溢出是指堆棧達(dá)到大值后,應(yīng)用內(nèi)存無(wú)法滿足需求了,導(dǎo)致Crash現(xiàn)象,這是一個(gè)結(jié)果。舉例:還是上文中的那個(gè)人,他想買(mǎi)300萬(wàn)的房子,這樣根本就買(mǎi)不了。
如果說(shuō)上面的例子還不足以理解其原因,下面再舉一例:
如上例所示,delay的時(shí)間過(guò)長(zhǎng)也可能導(dǎo)致資源無(wú)法及時(shí)釋放,這也是一種形式的內(nèi)存泄露。
一次內(nèi)存泄露可能并不能有實(shí)質(zhì)性的結(jié)果,但是不好的編程習(xí)慣,多次的內(nèi)存泄露終會(huì)導(dǎo)致內(nèi)存溢出,引發(fā)重大的bug,這是我們應(yīng)該極力避免的事情。
雖然在Android開(kāi)發(fā)中Java是基礎(chǔ)語(yǔ)法,Java的垃圾回收機(jī)制比較智能,很多情況下并不需要編程者去思考很多問(wèn)題,但是正是因?yàn)檫@種機(jī)制可能會(huì)導(dǎo)致很多不可預(yù)知的小錯(cuò)誤的累積。
當(dāng)然,導(dǎo)致內(nèi)存泄露和溢出的問(wèn)題有很多,比如在開(kāi)發(fā)中由于圖片資源占用過(guò)多。這些問(wèn)題我會(huì)在今后的博文中跟大家繼續(xù)分享。我個(gè)人也在不斷的學(xué)習(xí)之中,希望各位讀者能與我也做一些交流探討。
一個(gè)優(yōu)秀的開(kāi)發(fā)者不僅僅應(yīng)該局限于寫(xiě)出程序,而是寫(xiě)好程序。