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