「性能優(yōu)化1.2」異步優(yōu)化

「性能優(yōu)化1.0」啟動分類及啟動時間的測量
「性能優(yōu)化1.1」計算方法的執(zhí)行時間
「性能優(yōu)化1.2」異步優(yōu)化

一嘹吨、異步優(yōu)化

在上一小節(jié)中卵凑,我通過獲取應用的啟動時間和每一個方法執(zhí)行之間之后萌庆,我們發(fā)現(xiàn),如果在 Application 或者 MainActivity 生命周期中串行去執(zhí)行這些第三方庫的初始化,是會拖慢整個應用的啟動過程的,因此我們想通過子線程與主線程并行的方式來分擔主線程的工作猜惋,從而減少主線程的執(zhí)行時間。

并行執(zhí)行

1.1培愁、讓任務執(zhí)行在子線程中

1.1.1著摔、常規(guī)方案

我們常規(guī)的方式是怎樣的呢?

public void onCreate(){
    new Thread() {
        public run() {
            //執(zhí)行任務1
            //執(zhí)行任務2
            //執(zhí)行任務3
        }
    }.start();
}

但是這樣是不優(yōu)雅的竭钝,首先直接 new Thread() 這種方式比較簡單粗暴梨撞,而且這里只是開啟一個線程,我們最初的想法是想每一個異步任務就使用一個線程去執(zhí)行香罐。那么我們的偽代碼就變成如下這種方式:

public void onCreate(){
    new Thread() {
        public run() {
            //執(zhí)行任務1
        }
    }.start();
    
    new Thread() {
        public run() {
            //執(zhí)行任務2
        }
    }.start();
    
    new Thread() {
        public run() {
            //執(zhí)行任務3
        }
    }.start();
}

那要多個線程卧波,那我就創(chuàng)建多個線程唄,當然這種方式確實要比第一種好一些庇茫,因為它可以更加充分地利用 CPU 港粱,但是直接創(chuàng)建線程還是不優(yōu)雅,所以使用線程池來管理這些線程會好一些旦签。

1.1.2查坪、線程池管理

通過以下方式就可以獲取到我們對應的線程池,但是這個線程個數(shù)不能隨意填宁炫,我們要能充分利用到 CPU 資源偿曙,因此我們可以參考 AsyncTask 它是如何去設置核心線程數(shù)的。

Executors service = Executors.newFixedThreadPool(核心線程個數(shù));

AsyncTask 設置核心線程數(shù)

private static final int CPU_COUNT = Runtime.getRuntime().availableProcessors();
//CORE_POOL_SIZE 就是核心線程數(shù)
private static final int CORE_POOL_SIZE = Math.max(2, Math.min(CPU_COUNT - 1, 4));

因此有了這個核心線程數(shù)之后我們的代碼就變成如下方式:

private static final int CPU_COUNT = Runtime.getRuntime().availableProcessors();
private static final int CORE_POOL_SIZE = Math.max(2, Math.min(CPU_COUNT - 1, 4));
//參考AsyncTask來設置線程的個數(shù)羔巢。
ExecutorService service = Executors.newFixedThreadPool(CORE_POOL_SIZE);

關于核心線程數(shù)的設置是一個比較小的知識點望忆,不過這個對 CPU 資源的利用是很有幫助的。

通過改造后竿秆,我們現(xiàn)在的代碼如下:

@Override
public void onCreate() {
    super.onCreate();
    //參考AsyncTask來設置線程的個數(shù)启摄。
    ExecutorService service = Executors.newFixedThreadPool(CORE_POOL_SIZE);
    service.submit(new Runnable() {
        @Override
        public void run() {
            initBugly();
        }
    });
    service.submit(new Runnable() {
        @Override
        public void run() {
            initImageLoader();
        }
    });
}

還記得我們來上一小節(jié)中使用 AOP 來計算每一個方法的耗時,那么現(xiàn)在我們來對比一下通過異步加載和沒有異步加載這兩種方式的時間差別幽钢。

  • 沒有異步加載的代碼執(zhí)行結果
2019-03-17 20:29:12.946 10094-10094/com.example.perfermance E/PerformanceAop: method MyApplication.attachBaseContext(..) cost:1
2019-03-17 20:29:12.979 10094-10094/com.example.perfermance E/PerformanceAop: method MyApplication.initBugly() cost:12
2019-03-17 20:29:13.002 10094-10094/com.example.perfermance E/PerformanceAop: method MyApplication.initImageLoader() cost:23
2019-03-17 20:29:13.002 10094-10094/com.example.perfermance E/PerformanceAop: method MyApplication.onCreate() cost:35
  • 異步加載的代碼執(zhí)行結果:
2019-03-17 22:07:38.022 13948-13948/com.example.perfermance E/PerformanceAop: method MyApplication.attachBaseContext(..) cost:1
2019-03-17 22:07:38.062 13948-13948/com.example.perfermance E/PerformanceAop: method MyApplication.onCreate() cost:3
2019-03-17 22:07:38.078 13948-13967/com.example.perfermance E/PerformanceAop: method MyApplication.initBugly() cost:15
2019-03-17 22:07:38.094 13948-13968/com.example.perfermance E/PerformanceAop: method MyApplication.initImageLoader() cost:28

通過兩次輸出的 log 數(shù)據(jù)對比歉备,可以看出主線程執(zhí)行的 onCreate 方法的執(zhí)行時間從原來的 35ms 減到到了 3ms 。

但是這里又有另外一個問題匪燕,那就是有一些方法是必須在 Application onCreate 執(zhí)行完成之前完成初始化的蕾羊,因為在 MainActivity 中就需要使用到,那我們上面的異步就會有問題了帽驯,那如何解決這個問題呢肚豺?

1.1.3、異步任務必須在某一個階段執(zhí)行完成

我們還是以 initBugly() 方法來舉例界拦,這個方法是在異步線程中執(zhí)行吸申,如何控制讓其在 Application onCreate 執(zhí)行完畢之前它先完成呢?

這時就需要使用到 CountDownLatch 了,我們先來看看示例圖:

異步任務必須在某一個階段執(zhí)行完成
  • 定義一個 CountDownLatch
//Application
private CountDownLatch countDownLatch = new CountDownLatch(1);
  • 在方法執(zhí)行完畢時截碴,執(zhí)行 countDownLatch.countDown()
private void initBugly() {
    try {
        //模擬initBugly耗時
        Thread.sleep(3000);
    } catch (Exception e) {
        e.printStackTrace();
    }
    Log.e(TAG, "初始化initBugly完畢");
    //數(shù)量減一
    countDownLatch.countDown();
}
  • 等待 countDownLatch.await()

在 onCreate 方法結束點等待梳侨,如果在此處之前之前調用了countDownLatch.countDown(),那么就直接跳過日丹,否則就在此等待走哺。

public void onCreate() {
    super.onCreate();
    //參考AsyncTask來設置線程的個數(shù)。
    ExecutorService service = Executors.newFixedThreadPool(CORE_POOL_SIZE);
    service.submit(new Runnable() {
        @Override
        public void run() {
            initBugly();
        }
    });
    service.submit(new Runnable() {
        @Override
        public void run() {
            initImageLoader();
        }
    });
    
    
    //在 onCreate 方法中等待哲虾,如果在此處之前之前調用了countDownLatch.countDown()丙躏,那么就直接跳過,否則就在此等待束凑。
    try {
        countDownLatch.await();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    Log.e(TAG, "Application onCreate 執(zhí)行完畢");
}

這樣晒旅,我們的 Application onCreate 方法就會等待異步任務 initBugly 執(zhí)行完畢之后才會結束 onCreate 這個方法的生命周期。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末汪诉,一起剝皮案震驚了整個濱河市废恋,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌扒寄,老刑警劉巖鱼鼓,帶你破解...
    沈念sama閱讀 222,252評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異该编,居然都是意外死亡迄本,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評論 3 399
  • 文/潘曉璐 我一進店門课竣,熙熙樓的掌柜王于貴愁眉苦臉地迎上來岸梨,“玉大人,你說我怎么就攤上這事稠氮。” “怎么了半开?”我有些...
    開封第一講書人閱讀 168,814評論 0 361
  • 文/不壞的土叔 我叫張陵隔披,是天一觀的道長。 經常有香客問我寂拆,道長奢米,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,869評論 1 299
  • 正文 為了忘掉前任纠永,我火速辦了婚禮鬓长,結果婚禮上,老公的妹妹穿的比我還像新娘尝江。我一直安慰自己涉波,他們只是感情好,可當我...
    茶點故事閱讀 68,888評論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著啤覆,像睡著了一般苍日。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上窗声,一...
    開封第一講書人閱讀 52,475評論 1 312
  • 那天相恃,我揣著相機與錄音,去河邊找鬼笨觅。 笑死拦耐,一個胖子當著我的面吹牛,可吹牛的內容都是我干的见剩。 我是一名探鬼主播杀糯,決...
    沈念sama閱讀 41,010評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼炮温!你這毒婦竟也來了火脉?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,924評論 0 277
  • 序言:老撾萬榮一對情侶失蹤柒啤,失蹤者是張志新(化名)和其女友劉穎倦挂,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體担巩,經...
    沈念sama閱讀 46,469評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡方援,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 38,552評論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了涛癌。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片犯戏。...
    茶點故事閱讀 40,680評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖拳话,靈堂內的尸體忽然破棺而出先匪,到底是詐尸還是另有隱情,我是刑警寧澤弃衍,帶...
    沈念sama閱讀 36,362評論 5 351
  • 正文 年R本政府宣布呀非,位于F島的核電站,受9級特大地震影響镜盯,放射性物質發(fā)生泄漏岸裙。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,037評論 3 335
  • 文/蒙蒙 一速缆、第九天 我趴在偏房一處隱蔽的房頂上張望降允。 院中可真熱鬧,春花似錦艺糜、人聲如沸剧董。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽送滞。三九已至侠草,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間犁嗅,已是汗流浹背边涕。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留褂微,地道東北人功蜓。 一個月前我還...
    沈念sama閱讀 49,099評論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像宠蚂,于是被迫代替她去往敵國和親式撼。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,691評論 2 361

推薦閱讀更多精彩內容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴謹 對...
    cosWriter閱讀 11,113評論 1 32
  • 所有知識點已整理成app app下載地址 J2EE 部分: 1.Switch能否用string做參數(shù)求厕? 在 Jav...
    侯蛋蛋_閱讀 2,450評論 1 4
  • 第十章:Android的消息機制 Handler是Android消息機制的上層接口著隆,開發(fā)人員只需要與它交互即可,底...
    loneyzhou閱讀 682評論 0 1
  • 2.1 Activity 2.1.1 Activity的生命周期全面分析 典型情況下的生命周期:在用戶參與的情況下...
    AndroidMaster閱讀 3,051評論 0 8
  • 正與生命里的一切相同 我們愛得太是匆匆 好像只是昨天 彼此叫著專屬于對方的名字 我總是說呀癣,怎么又下雨了美浦? 而你總是...
    若惜惜閱讀 243評論 0 0