讓普通 Java 類自動感知 Activity Lifecycle

《億級 Android 架構》 地址:https://xiaozhuanlan.com/AndroidArch

背景

在 Android 開發(fā)中,我們都很熟悉 Activity 的 Lifecycle,并且會在特定的 Lifecycle 下執(zhí)行特定的操作。當然,我們清楚 Lifecycle 本身是帶有 Android 特質的,那嘗試設想下,如果普通的 Java Class 也能自動感知 Lifecycle 呢?咋一聽這個想法似乎背后意義不大,但在實際探索中,我們發(fā)現這個特性能為我們達成一些之前未考慮到或者不易實現的優(yōu)化。

本文分享下我們基于這個思想所開發(fā)的框架:AutoLifecycle 及其帶來的一些有意思的實踐。

  • 優(yōu)化一:當Activity進入onDestroy時,自動取消網絡請求返回
  • 優(yōu)化二:自動將網絡請求時機提前到View渲染之前,提高頁面打開速度
  • 優(yōu)化三:MVP改進,讓Presenter和View自動bind/unBind

注:下文提到的Lifecycle-Aware就是這里指代的讓普通 Java Class 自動獲取 Lifecycle。

實踐及優(yōu)化

優(yōu)化一:當Activity進入onDestroy時,自動取消網絡請求返回

在網絡請求時,相信大家都有一個經驗:在每個網絡結果回來時,我們做的第一件事不是顯示數據,而是寫個if-else判斷Activity還在不在。

mTopApiObservable
  ...
  .subscribe(new Subscriber<Object>() {
      @Override
      public void onNext(Object data) {
        if(activity == null) {
            return; // 判斷Activity是否還在,不在就不去顯示數據
        }
        
        display(data); // 顯示數據
      }
      ...
  });

由于網絡請求都是異步的,所以不得不做這樣的判斷,來防止不可預測的空指針問題或內存泄漏問題。

既然你總是擔心Activity還在不在,那么如果我們通過Lifecycle-Aware讓每個網絡請求能自動感知Activity的onDestroy事件
并在onDestroy時,自動把網絡請求結果取消掉不再返回,那就能夠消除這個擔憂了。

mTopApiObservable
  ...
  .compose(bindUntilEvent(ActivityLifecycle.DESTROY)) // 綁定Activity的onDestroy事件
  .subscribe(new Subscriber<Object>() {
      @Override
      public void onNext(Object data) {
        display(data); // 直接去顯示數據
      }
      ...
  });

其中最關鍵的就是compose(bindUntilEvent(ActivityLifecycle.DESTROY))這句,它能達到的效果是:一旦Activity發(fā)生onDestroy時,Observer的數據就會停止向Subscriber里流動。從而保證onNext無需擔心ActivityDestroy這種情況。

在上面網絡請求的實踐里,你還可以根據自己的情況把Destroy換成Stop/Pause等,而且可以看出,這種自動取消機制可適用于任何Observable,不僅僅是網絡請求。

優(yōu)化二:自動將網絡請求提前到View Inflate之前,加速頁面渲染

先說下這項優(yōu)化的原理。
通常,我們會在ActivityonCreate里依次執(zhí)行下面的代碼:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.XXX);   // Inflate View
    findViewByIds();                // 初始化View
    presenter.loadDataFromServer(); // 發(fā)起網絡請求
}

即在Inflate View初始化View之后,才發(fā)起網絡請求去加載數據。

而實際上,網絡請求是不占用主線程的,如果能在Inflate View之前就在其他線程發(fā)起網絡請求,可以把整個頁面顯示耗時縮短100ms-200ms

LoadBeforeInflate優(yōu)化效果 (1).png

現在有了AutoLifecycle框架,我們就可以很輕松實現:讓Presenter自動監(jiān)聽Inflate View這個生命周期,在那時發(fā)起網絡請求即可。

public class NewPresenter {

    public NewPresenter(IView iView) {
        ...
        // 向AutoLifecycle注冊
        AutoLifecycle.getInstance().init(this); 
    }

    // 當Activity Inflate View前自動回調
    @AutoLifecycleEvent(activity = ActivityLifecycle.PRE_INFLATE)
    private void onHostPreInflate() {
         loadDataFromServer(); // 發(fā)起網絡請求
    }
    ...
}

此時,我們的Activity也不用手動調用presenter.loadDataFromServer();了,因為Presenter內會在感知到Inflate View事件時自動發(fā)起網絡請求。

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.XXX);
    findViewByIds();
    // 無需手動啟動網絡請求
}

經過測試,在保證單個網絡請求耗時相同的情況下,頁面從onCreate顯示數據的渲染耗時可以從550ms縮短到367ms,也就是30%-40%的優(yōu)化,效果是非常不錯的,而且代碼也更加簡潔清晰。

優(yōu)化前

優(yōu)化后
[圖片上傳失敗...(image-889271-1510146010290)]

通過簡單的注冊AutoLifecycle,Presenter能夠自動感知到所有Lifecycle,甚至包括自定義的特殊Lifecycle,如下圖:
[圖片上傳失敗...(image-8c0f0b-1510146010290)]

B2.png

優(yōu)化三:MVP改進,讓Presenter和View自動bind/unBind

第一項優(yōu)化比較直接,可以先讓大家形成一個直觀印象。
我們項目是采用MVP項目,對于Presenter的使用存在一段固定代碼,即在onCreate時調用bindView(),在onDestroy時調用unBindView()。如下圖:

public class OldActivity extends BaseActivity {

    BasePresenter mPresenter = new BasePresenter();

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mPresenter.bindView(this); // onCreate時手動bind Presenter 和 IView
    }

    @Override
    protected void onDestroy() {
        mPresenter.unbindView(); // onDestroy時手動unBindView
        super.onDestroy();
    }
}

那么,既然我們現在能讓一個普通類自動感知Lifecycle,那其實也就能讓Presenter在感知到onCreate自動bindView,在感知到onDestroy自動unBindView。
改進后的代碼如下:

public class NewActivity extends BaseActivity {

    NewPresenter mPresenter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mPresenter = new NewPresenter(this); // 只需要創(chuàng)建即可
    }
}

public class NewPresenter {

    private IView mIView;

    public NewPresenter(IView iView) {
        this.mIView = iView;
        // 向AutoLifecycle注冊即可獲得Lifecycle回調
        AutoLifecycle.getInstance().init(this); 
    }

    // 當Activity進入onCreate后自動調用
    @AutoLifecycleEvent(activity = ActivityLifecycle.CREATE)
    private void onHostCreate() {
        bindView(mIView); 
    }

    // 當Activity進入onDestroy后自動調用
    @AutoLifecycleEvent(activity = ActivityLifecycle.DESTROY)
    private void onHostDestroy() {
        unBindView();
    }
}

其實,在大家的平常開發(fā)中,還會存在許多類似Presenter的類:需要在某個特定的Lifecycle下執(zhí)行一些動作。這時就可以基于Lifecycle-Aware來讓這個普通類自動去執(zhí)行,而不是去每個Activity/Fragment里寫一遍,提高類的內聚性。

AutoLifecycle的核心原理

(TL;DR)
下面介紹下AutoLifecycle的關鍵實現部分,感興趣的讀者可以參考。

1. 讓Activity對外發(fā)送Lifecycle事件

使用過RxJava的同學知道里面有一個PublishSubject,基于觀察者模式,主動發(fā)送并接受消息。這里我們用PublishSubject來發(fā)送Lifecycle事件。見如下:

D1.png

這里的Lifecycle事件可以自己定義,比如前面提到的PRE_INFLATE事件,是在setContentView之前發(fā)送,類似:

D2.png

2. 感知某個Lifecycle的發(fā)生并自動執(zhí)行回調

上面提了,PublishSubject不僅能發(fā)送消息,還能接受自己的消息。基于這個特點,我們便可以監(jiān)聽每一個LifecycleEvent。如下圖:

D3.png

這里的參數Observable是我們希望被回調的函數,IContextLifecycle是指定的Lifecycle。即當指定的Lifecycle Event發(fā)生時,會自動subscribe提供的Observable。

基于這個功能,便可以實現上面場景一和場景二里的@AutoLifecycleEvent注解了,即把@AutoLifecycleEvent標注的函數包裝成一個Observable,通過這個executeOn來注冊函數的執(zhí)行生命周期即可。

3. 監(jiān)聽Lifecycle并取消網絡請求結果

在場景三里,我們?yōu)榫W絡請求的Observable提供了一個Transformer,它能在監(jiān)聽到某個Lifecycle發(fā)生時,停止數據流的向下流動。該Transformer的核心實現是:

D4.png

可以看出,當指定的Lifecycle一旦發(fā)生,我們網絡請求Observable就會停止向下傳遞數據。

4. 支持自定義Lifecycle,支持Activity/Fragment/DialogFrament等

可以看出,AutoLifecycle除了支持常規(guī)的生命周期,還能支持自定義的特殊生命周期,比如View Inflate前。

另外,上面都是以Activity為例,不過顯然這套框架可以靈活擴展,不局限于Activity,還能適用于Fragment、DialogFrament等。

總結

Lifecycle-Aware思想是Google官方提出來的概念:賦予普通類自動感知生命周期的能力。而本文也是基于這個思想,提供了一些具體實踐和優(yōu)化的思路,讀者同學可以根據自己的情況做更多的改進和嘗試。

——————
wingjay
謝謝。

參考

https://developer.android.com/topic/libraries/architecture/lifecycle.html
https://www.atatech.org/articles/63098
https://github.com/trello/RxLifecycle
http://reactivex.io/RxJava/javadoc/rx/subjects/PublishSubject.html
http://reactivex.io/RxJava/javadoc/rx/Observable.Transformer.html

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,326評論 25 708
  • 用兩張圖告訴你,為什么你的 App 會卡頓? - Android - 掘金 Cover 有什么料? 從這篇文章中你...
    hw1212閱讀 14,110評論 2 59
  • 懶得處理樣式了, 將就著看吧. 官網地址: https://developer.android.com/topic...
    Reddington_604e閱讀 1,923評論 0 1
  • 哪種葉子的黃最能打動你,對我來說,非銀杏莫屬了。是因為它在冬日獨占芳華,是因為它是韓劇浪漫色彩的主背景,說不清道不...
    木木夕_73f6閱讀 647評論 0 3
  • 今天發(fā)現一個真相——原來我值得這世界上所有美好的東西。 今天的年終獎對我沖擊力太大了,我第一次感受到我愛錢寶寶。我...
    清理放下閱讀 255評論 0 0

友情鏈接更多精彩內容