<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    demibug

      BlogJava :: 首頁 :: 聯(lián)系 :: 聚合  :: 管理
      24 Posts :: 3 Stories :: 2 Comments :: 0 Trackbacks
    最近在看程序的時候,看到好多地方都用到了Context,下面來學(xué)習(xí)一下:
    ACOD上關(guān)于Context的解釋:
    Interface to global information about an application environment. This is an abstract class whose implementation is provided by the Android system(added by me:By subclass). It allows access to application-specific resources and classes, as well as up-calls for application-level operations such as launching activities, broadcasting and receiving intents, etc.
     
    Context字面意思上下文,位于framework package的android.content.Context中,其實該類為LONG型,類似Win32中的Handle句柄,很多方法需要通過 Context才能識別調(diào)用者的實例,比如說Toast的第一個參數(shù)就是Context,一般在Activity中我們直接用this代替,代表調(diào)用者的 實例為Activity,而到了一個button的onClick(View view)等方法時,我們用this時就會報錯,所以我們可能使用ActivityName.this來解決,主要原因是因為實現(xiàn)Context的類主要有Android特有的幾個模型,Activity、Service以及BroadcastReceiver。
     
    兩種類型的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等。
     
    內(nèi)存泄露
    這樣如果context發(fā)生內(nèi)存泄露的話,就會泄露很多內(nèi)存。這里泄露的意思是gc沒有辦法回收activity的內(nèi)存。
     
    注釋:為什么GC沒有辦法回收相應(yīng)的內(nèi)存,個人感覺是因為傳遞Context會增加對象指針的引用計數(shù),所以基于智能指針技術(shù)的GC無法釋放相應(yīng)的內(nèi)存。
     
    當(dāng)屏幕旋轉(zhuǎn)的時候,系統(tǒng)會銷毀當(dāng)前的activity,保存狀態(tài)信息,再創(chuàng)建一個新的。比如我們寫了一個應(yīng)用程序,它需要加載一個很大的圖片,我們不希望每次旋轉(zhuǎn)屏幕的時候都銷毀這個圖片,重新加載。實現(xiàn)這個要求的簡單想法就是定義一個靜態(tài)的Drawable,這樣Activity 類創(chuàng)建銷毀它始終保存在內(nèi)存中。實現(xiàn)類似:

    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);
    }
    }
    這段程序看起來很簡單,但是卻問題很大。當(dāng)屏幕旋轉(zhuǎn)的時候會有l(wèi)eak(即gc沒法銷毀activity)。我們剛才說過,屏幕旋轉(zhuǎn)的時候系統(tǒng)會銷毀當(dāng)前的activity。但是當(dāng)drawable和view關(guān)聯(lián)后,drawable保存了view的 reference,即sBackground保存了label的引用,而label保存了activity的引用。既然drawable不能銷毀,它所引用和間接引用的都不能銷毀,這樣系統(tǒng)就沒有辦法銷毀當(dāng)前的activity,于是造成了內(nèi)存泄露。gc對這種類型的內(nèi)存泄露是無能為力的。避免這種內(nèi)存泄露的方法是避免activity中的任何對象的生命周期長過activity,避免由于對象對 activity的引用導(dǎo)致activity不能正常被銷毀。
     
    為了防止內(nèi)存泄露,我們應(yīng)該注意以下幾點:
    1. 不要讓生命周期長的對象引用activity context,即保證引用activity的對象要與activity本身生命周期是一樣的
    2. 對于生命周期長的對象,可以使用application context
    3. 避免非靜態(tài)的內(nèi)部類,盡量使用靜態(tài)類,避免生命周期問題,注意內(nèi)部類對外部對象引用導(dǎo)致的生命周期變化
    application context
    我們可以使用application context。application context伴隨application的一生,與activity的生命周期無關(guān)。application context可以通過Context.getApplicationContext或者Activity.getApplication方法獲取。
     
    Java里面通常是用一個static的變量(例如singleton之類的)來同步activity之間(程序里面類之間)的狀態(tài)。在android里面比較靠譜的做法是用application context來關(guān)聯(lián)這些狀態(tài)。
    每個activity都是context,里面包含了運行時的狀態(tài)。同樣application也有一個context,android會保證這個context是唯一的實例。
    做一個你自己的application context需要繼承android.app.Application,然后在app的manifest里面說明這個類。android會自動幫你創(chuàng)建你這個類的實例,接著你用Context.getApplicationContext()方法就能在各個activity里
    面獲得這個application context了。

     class MyApp extends Application {

      private String myState;

      public String getState(){
        return myState;
      }
      public void setState(String s){
        myState = s;
      }
    }

    class Blah extends Activity {

      @Override
      public void onCreate(Bundle b){
        ...
        MyApp appState = ((MyApp)getApplicationContext());
        String state = appState.getState();
        ...
      }

    posted on 2011-11-10 23:19 Hiji 閱讀(466) 評論(0)  編輯  收藏

    只有注冊用戶登錄后才能發(fā)表評論。


    網(wǎng)站導(dǎo)航:
     
    主站蜘蛛池模板: 日韩欧美亚洲国产精品字幕久久久| 国产jizzjizz视频免费看| 亚洲精品乱码久久久久久按摩| 亚洲a∨无码精品色午夜| 久久不见久久见中文字幕免费 | 亚洲成人免费电影| 在线观看亚洲人成网站| 99re在线精品视频免费| 亚洲国产综合精品中文第一区| 青青操免费在线视频| 亚洲成a人片在线观看无码| 久久久免费的精品| 亚洲精品乱码久久久久久下载 | 久久精品国产精品亚洲色婷婷| 免费网站看av片| 亚洲精品线在线观看| 国产大片91精品免费观看不卡| 亚洲人成网站看在线播放| 成人免费毛片观看| 免费一区二区三区在线视频 | 免费无码av片在线观看| 亚洲首页在线观看| 国产精品久久久久久久久久免费| 亚洲人成无码网站在线观看| 五月天婷亚洲天综合网精品偷| 精品一区二区三区免费观看| 亚洲av中文无码乱人伦在线r▽| 最近中文字幕大全免费视频| 亚洲一本到无码av中文字幕| 亚洲成a人无码av波多野按摩| 免费无码又爽又刺激网站| 亚洲av产在线精品亚洲第一站| 国产精品视_精品国产免费| 久久国产一片免费观看| 亚洲蜜芽在线精品一区| 日本免费网站观看| a毛片在线免费观看| 国产成人精品亚洲日本在线| 亚洲日韩中文在线精品第一| 2019中文字幕免费电影在线播放| 无码天堂亚洲国产AV|