2014年5月8日 星期四

Android Context理解

在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 異常.







沒有留言:

張貼留言