在android中context可以作很多操作,但是最主要的功能是加載和訪問資源。在android中有兩種
context,一種是 application context,一種是activity context,通常我們在各種類和方法間
傳遞的是activity context。
比如一個activity的onCreate:
protected
void
onCreate(Bundle state) {
super.onCreate(state);
TextView label =
new
TextView(
this
);
//傳遞context給view control
label.setText(
"Leaks are bad"
);
setContentView(label);
}
把activity context傳遞給view,意味著view擁有一個指向activity的引用,進而引用activity佔
有的資源:view hierachy, resource等。
這樣如果context發生內存洩露的話,就會洩露很多內存。
這裡洩露的意思是gc沒有辦法回收activity的內存。
Leaking an entire activity是很容易的一件事。
當屏幕旋轉的時候,系統會銷毀當前的activity,保存狀態信息,再創建一個新的。
比如我們寫了一個應用程序,它需要加載一個很大的圖片,我們不希望每次旋轉屏 幕的時候都銷毀這個
圖片,重新加載。實現這個要求的簡單想法就是定義一個靜態的Drawable,這樣Activity 類創建銷毀
它始終保存在內存中。
實現類似:
public
class
myactivity extends Activity {
private
static
Drawable sBackground;
protected
void
onCreate(Bundle state) {
super.onCreate(state);
TextView label =
new
TextView(
this
);
label.setText(
"Leaks are bad"
);
if
(sBackground == null) {
sBackground = getDrawable(R.drawable.large_bitmap);
}
label.setBackgroundDrawable(sBackground);
//drawable attached to a view
setContentView(label);
}
}
這段程序看起來很簡單,但是卻問題很大。當屏幕旋轉的時候會有leak(即gc沒法銷毀activity)。
我們剛才說過,屏幕旋轉的時候系統會銷毀當前的activity。但是當drawable和view關聯後,drawable
保存了view的 reference,即sBackground保存了label的引用,而label保存了activity的引用。
既然drawable不能銷毀,它所 引用和間接引用的都不能銷毀,這樣系統就沒有辦法銷毀當前的activity,
於是造成了內存洩露。gc對這種類型的內存洩露是無能為力的。
避免這種內存洩露的方法是避免activity中的任何對象的生命週期長過activity,避免由於對象對
activity的引用導致activity不能正常被銷毀。我們可以使用application context。application
context伴隨application的一生,與activity的生命週期無關。application context可以通過
Context.getApplicationContext或者Activity.getApplication方法獲取。
避免context相關的內存洩露,記住以下幾點:
1. 不要讓生命週期長的對象引用activity context,即保證引用activity的對象要與activity本身
生命週期是一樣的
2. 對於生命週期長的對象,可以使用application context
3. 避免非靜態的內部類,儘量使用靜態類,避免生命週期問題,注意內部類對外部對象引用導致的生命
週期變化
根據packageName構造Context
通常情況下獲取當前應用的context的方法是getApplicationContext,
但是通過根據其他的packageName如何構造 Context呢?
Android平台的應用實例其實還可以通過其他方式構造。
比如代碼
try
{
Context ctx= createPackageContext("com.android123.Cwj", 0);
//ctx已經是com.android123.cwj的實例
}
catch (NameNotFoundException e)
{
//可能由於 pacakgeName不存在所以必須處理該異常
}
複製代碼需要注意的是,createPackageContext方法的第二個參數可選為CONTEXT_INCLUDE_CODE 和 CONTEXT_IGNORE_SECURITY ,
定義分別為4和2,上面為0。一般忽略安全錯誤問題可以通過CONTEXT_IGNORE_SECURITY 標記,
同時可能還需要處理 SecurityException 異常.
沒有留言:
張貼留言