[Android]使用Dagger 2依賴註入 - 圖表創建的性能(翻譯)

来源:http://www.cnblogs.com/tiantianbyconan/archive/2016/01/04/5098943.html
-Advertisement-
Play Games

<font color=" ff0000" <strong 以下內容為原創,歡迎轉載,轉載請註明來自天天博客:<http://www.cnblogs.com/tiantianbyconan/p/5098943.html </strong </font 使用Dagger 2依賴註入 圖表創建的性能 ....



以下內容為原創,歡迎轉載,轉載請註明
來自天天博客:http://www.cnblogs.com/tiantianbyconan/p/5098943.html

使用Dagger 2依賴註入 - 圖表創建的性能

原文:http://frogermcs.github.io/dagger-graph-creation-performance/

#PerfMatters - 最近非常流行標簽,尤其在Android世界中。不管怎樣,apps只需要正常工作就可以的時代已經過去了。現在所有的一切都應該是令人愉悅的,流暢並且快速。舉個例子,Instagram 花費了半年的時間 只是讓app更加快速,更加美觀,和更好的屏幕適配性。

這就是為什麼今天我想去分享給你一些小的建議,它會在你app啟動時間上有很大的影響(尤其是當app使用了一些額外庫的時候)。

對象圖表的創建

大多情況下,在app開發過程中,它的啟動時間或多或少會增加。有時隨著一天天地開發它是很難被註意到的,但是當你把第一個版本和你能找到的最近的版本比較時區別就會相對比較大了。

原因很可能就在於dagger對象圖表的創建過程。

Dagger 2?你可能會問,確切地說 - 就算你移除了那些基於反射的實現方案,並且你的代碼是在編譯時期生成的,但是別忘了對象的創建仍然發生是在運行時。

對象(還有它的依賴)會在第一次被註入時創建。Jake Wharton 在Dagger 2演示中的一些幻燈片很清楚地展示了這一點:

以下表示在我們的 GithubClient 例子app中它是怎樣的:

  1. App第一次(被kill之後)被啟動。Application對象並沒有@Inject屬性,所以只有AppComponent對象被創建。
  2. App創建了SplashActivity - 它有兩個@Inject屬性:AnalyticsManagerSplashActivityPresenter
  3. AnalyticsManager依賴已被創建的Application對象。所以只有AnalyticsManager構造方法被調用。
  4. SplashSctivityPresenter依賴:SplashActivityValidatorUserManagerSplashActivity已被提供,ValidatorUserManager應該被創建。
  5. UserManager依賴需要被創建的GithubApiService。之後UserManager被創建。
  6. 現在我們擁有了所有依賴,SplashActivityPresenter被創建。

有點混亂,但是就結果來說,在SplashActivity被創建之前(我們假設對象註入的操作只會在onCreate()方法中執行)我們必須要等待以下構造方法(可選配置):

  • GithubApiService(它也使用了一些依賴,如OkHttpClient,一個RestAdapter
  • UserManager
  • Validator
  • SplashActivityPresenter
  • AnalyticsManager

一個接一個地被創建。

嘿,別擔心,更複雜地圖表也幾乎被立即創建。

問題

現在讓我們想象下,我們有兩個外部的庫需要在app啟動時被初始化(比如,Crashlytics, Mixpanel, Google Analytics, Parse等等)。想象下我們的HeavyExternalLibrary看起來如下:

public class HeavyExternalLibrary {

    private boolean initialized = false;

    public HeavyExternalLibrary() {
    }

    public void init() {
        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        initialized = true;
    }

    public void callMethod() {
        if (!initialized) throw new RuntimeException("Call init() before you use this library");
    }
}

簡單說 - 構造方法是空的,並且調用幾乎不花費任何東西。但是有一個init()方法,它耗時500ms並且在我們使用這個庫之前必須要被調用。確保在我們module的某處的某一時刻調用了init()

//AppModule

@Provides
@Singleton
HeavyExternalLibrary provideHeavyExternalLibrary() {
    HeavyExternalLibrary heavyExternalLibrary = new HeavyExternalLibrary();
    heavyExternalLibrary.init();
    return heavyExternalLibrary;
}

現在我們的HeavyExternalLibrary成為了SplashActivityPresenter的一部分:

@Provides
@ActivityScope
SplashActivityPresenter
provideSplashActivityPresenter(Validator validator, UserManager userManager, HeavyExternalLibrary heavyExternalLibrary) {
    return new SplashActivityPresenter(splashActivity, validator, userManager, heavyExternalLibrary);
}

然後會發生什麼?我們app啟動時間需要500ms還多,只是因為HeavyExternalLibrary的初始化,這過程會在SplashActivityPresenter依賴圖表創建中執行。

測量

Android SDK(Android Studio本身)給我們提供了一個隨著應用執行的時間的可視化的工具 - Traceview。多虧這個我們可以看見每個方法花了多少時間,並且找出註入過程中的瓶頸。

順便說一下,如果你以前沒有見過它,可以在Udi Cohen的博客看下這篇Android性能優化相關的文章。

Traceview可以直接從Android Studio(Android Monitor tab -> CPU -> Start/Stop Method Tracing)啟動,它有時並不是那麼精確的,尤其是當我們嘗試在app啟動時點擊Start

對於我們而言,幸運的是當我們知道確切的需要被測量的代碼位置時,有一個可以使用的方法。Debug.startMethodTracing()可以用來指定我們代碼中需要被啟動測量的位置。Debug.stopMethodTracing()停止追蹤並且創建一個新的文件。

為了實踐,我們測量了SplashActivity的註入過程:

@Override
protected void setupActivityComponent() {
    Debug.startMethodTracing("SplashTrace");
    GithubClientApplication.get(this)
            .getAppComponent()
            .plus(new SplashActivityModule(this))
            .inject(this);
    Debug.stopMethodTracing();
}

setupActivityComponent()是在onCreate()中調用的。

文檔結果被保存在/sdcard/SplashTrace.trace中。

在你的terminal中把它pull出來:

$ adb pull /sdcard/SplashTrace.trace

現在閱讀這個文件所要做的全部事情只是把它拖拽到Android Studio:

你應該會看到類似以下的東西:

當然,我們這個例子中的結果是非常清晰的:AppModule_ProvideHeavyExternalLibraryFactory.get()(HeavyExternalLibrary被創建的地方)花費了500ms。

真正好玩的地方是,縮放trace尾部的那一小塊地方:

看到不同之處了嗎?比如構建類:AnalyticsManager花了小於1ms。

如果你想看到它,這裡有這個例子中的SplashTrace.trace文件。

解決方案

不幸的是,對於這類性能問題,有時並沒有明確的回答。這裡有兩種方式會給我們很大的幫助。

懶載入(臨時的解決方案)

首先,我們要思考是否你需要所有的註入依賴。也許其中一部分可以延遲一定時間後再載入?當然這並不解決真正的問題(UI線程將會在第一次調用Lazy<>.get()方法的時候阻塞)。但是在某些情況下對啟動耗時有幫助(尤其是很少地方會使用到的一些對象)。查看Lazy<>介面文檔獲取更多的信息和例子代碼。

簡單說,每一個你使用@Inject SomeClass someClass的地方都可以替換成@Inject Lazy<SomeClass> someClassLazy(構造方法註入也是)。然後獲取某個類的實例時必須要調用someClassLazy.get()

非同步對象創建

第二種選擇(它仍然只是更多的想法而不是最終的解決方案)是在後臺線程中的某處進行對象的初始化,緩存所有方法的調用併在初始化之後再回調它們。

這種方案的缺點是它必須要單獨地準備我們要包含的所有類。並且它只有在方法調用可以被執行的將來(就像任何的analytics - 在一些事件被髮生之後才可以),這些對象才可能正常工作。

以下就是我們的HeavyExternalLibrary使用這種解決方案後的樣子:

public class HeavyLibraryWrapper {

    private HeavyExternalLibrary heavyExternalLibrary;

    private boolean isInitialized = false;

    ConnectableObservable<HeavyExternalLibrary> initObservable;

    public HeavyLibraryWrapper() {
        initObservable = Observable.create(new Observable.OnSubscribe<HeavyExternalLibrary>() {
            @Override
            public void call(Subscriber<? super HeavyExternalLibrary> subscriber) {
                HeavyLibraryWrapper.this.heavyExternalLibrary = new HeavyExternalLibrary();
                HeavyLibraryWrapper.this.heavyExternalLibrary.init();
                subscriber.onNext(heavyExternalLibrary);
                subscriber.onCompleted();
            }
        }).subscribeOn(Schedulers.io()).observeOn(AndroidSchedulers.mainThread()).publish();

        initObservable.subscribe(new SimpleObserver<HeavyExternalLibrary>() {
            @Override
            public void onNext(HeavyExternalLibrary heavyExternalLibrary) {
                isInitialized = true;
            }
        });

        initObservable.connect();
    }

    public void callMethod() {
        if (isInitialized) {
            HeavyExternalLibrary.callMethod();
        } else {
            initObservable.subscribe(new SimpleObserver<HeavyExternalLibrary>() {
                @Override
                public void onNext(HeavyExternalLibrary heavyExternalLibrary) {
                    heavyExternalLibrary.callMethod();
                }
            });
        }
    }
}

HeavyLibraryWrapper構造方法被調用,庫的初始化會在後臺線程(這裡的Schedulers.io())中執行。在此期間,當用戶調用callMethod(),它會增加一個新的subscription到我們的初始化過程中。當它完成時(onNext()方法返回一個已初始化的HeavyExternalLibrary對象)被緩存的回調會被傳送到這個對象。

目前為止,這個想法還是非常簡單並且仍然是在開發之中。這裡可能會引起記憶體泄漏(比如,我們不得不在callMethod()方法中傳入一些參數),但一般還是適用於簡單的情況下的。

還有其它方案?

性能優化的過程是非常孤獨的。但是如果你想要分享你的ideas,請在這裡分享吧。

感謝你的閱讀!

代碼:

以上描述的完整代碼可見Github [repository]。

作者

Miroslaw Stanek

Head of Mobile Development @ Azimo


[Android]使用Dagger 2依賴註入 - DI介紹(翻譯):

http://www.cnblogs.com/tiantianbyconan/p/5092083.html


[Android]使用Dagger 2依賴註入 - API(翻譯):

http://www.cnblogs.com/tiantianbyconan/p/5092525.html


[Android]使用Dagger 2依賴註入 - 自定義Scope(翻譯):

http://www.cnblogs.com/tiantianbyconan/p/5095426.html


[Android]使用Dagger 2依賴註入 - 圖表創建的性能(翻譯):

http://www.cnblogs.com/tiantianbyconan/p/5098943.html


[Android]Dagger2Metrics - 測量DI圖表初始化的性能(翻譯):

http://www.cnblogs.com/tiantianbyconan/p/5193437.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 看完過後,你會學到:1學習IOS開發中的分類實現,2以及類方法的書寫,3以及字元串的MD5加密/解密.---------------------------wolfhous------------------[1]新建工程,新建分類[2]分類的命名方式以及類的選擇[3]書寫NSString分類的類方...
  • 1、錯誤代碼:No architectures to compile for (ONLY_ACTIVE_ARCH=YES, active arch=x86_64, VALID_ARCHS=armv7 armv7s)build 這個target的時候命令行報錯。可以在 build setting 中,...
  • 在上一節中,我們通過探討應用的系統設置的基本功能,瞭解運用bundle捆綁包以及plist文件的基本開發。用戶能夠使用設置應用來聲明他們的偏好設置,那麼我們怎樣去調用用戶所設置的參數呢,上節我們只創建了Setting.bundle捆綁包,在這個基礎上,這一節,我們將繼續討論,關於如何讀取應用中...
  • ViewPager類來自於支持庫。與fragment類不同,ViewPager只存在於支持庫中。而且,可以預見,即使在SDK的後續版本中,並不存在標準的ViewPager類。 《...
  • 本文翻譯自http://developer.android.com/intl/zh-cn/tools/building/multidex.html#about。主要介紹當我們Android App中函數超過65536時構建失敗的原因及解決辦法! ------------------------...
  • 1.前言 隨著移動支付時代的到來,TouchID 指紋驗證迅速被支付寶,微信錢包普及,相信各位朋友使用後也大呼方便。之前寫了篇關於iOS9的3D Touch的集成使用,有朋友在我博客下提到,讓我寫一篇關於指紋解鎖的blog,元旦結束後,立馬前來向大家分享我的一點點拙見。關於3D Touch 感興趣....
  • 大家都知道,WKWebview是沒有查看大圖的屬性或者方法的,所以只能通過js與之交互來實現這一功能,原理:通過js獲取頁面的圖片,把它存放到數組,給圖片添加點擊事件,通過index顯示大圖就行了其實很簡單,給WKWebview寫個類別,添加兩個方法就行了,然後在WKWebview的兩個協議中調用下...
  • 自定義UICollectionView,主要會用到以下幾個方法:- (void)prepareLayout; 第一次載入layout、刷新layout、以及- (BOOL)shouldInvalidateLayoutForBoundsChange:(CGRect)newBounds;這個方法返回ye...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...