IdleHandler的使用及原理

IdleHandler方式就是利用其特性,只有CPU空閑的時(shí)候才會(huì)執(zhí)行相關(guān)任務(wù)拔恰,并且我們可以分批進(jìn)行任務(wù)初始化,可以有效緩解界面的卡頓。

簡單用法代碼如下:

        Looper.myQueue().addIdleHandler(object: MessageQueue.IdleHandler {
            override fun queueIdle(): Boolean {
                //執(zhí)行任務(wù)
                return false;
            }
        })

可以將上述代碼添加到Activity onCreate中照宝,在queueIdle()方法中實(shí)現(xiàn)延遲執(zhí)行任務(wù),在主線程空閑句葵,也就是activity創(chuàng)建完成之后硫豆,它會(huì)執(zhí)行queueIdle()方法中的代碼。

如何設(shè)置是否重復(fù)執(zhí)行

queueIdle()返回true表示可以反復(fù)執(zhí)行該方法笼呆,即執(zhí)行后還可以再次執(zhí)行熊响;返回false表示執(zhí)行完該方法后會(huì)移除該IdleHandler,即只執(zhí)行一次诗赌。

IdleHandler源碼解析:

IdleHandler屬于MessageQueue內(nèi)部接口汗茄,只有一個(gè)queueIdle()方法聲明。通過方法addIdleHandler將我們的idleHandler添加到集合中铭若。

public final class MessageQueue {

    private final ArrayList<IdleHandler> mIdleHandlers = new ArrayList<IdleHandler>();
  
 
     /**
     * Add a new {@link IdleHandler} to this message queue.  This may be
     * removed automatically for you by returning false from
     * {@link IdleHandler#queueIdle IdleHandler.queueIdle()} when it is
     * invoked, or explicitly removing it with {@link #removeIdleHandler}.
     *
     * <p>This method is safe to call from any thread.
     *
     * @param handler The IdleHandler to be added.
     */
    public void addIdleHandler(@NonNull IdleHandler handler) {
        if (handler == null) {
            throw new NullPointerException("Can't add a null IdleHandler");
        }
        synchronized (this) {
            mIdleHandlers.add(handler);
        }
    }
 
 
    /**
     * Callback interface for discovering when a thread is going to block
     * waiting for more messages.
     */
    public static interface IdleHandler {
        boolean queueIdle();
    }
}
IdleHandler的queueIdle方法何時(shí)執(zhí)行

在MessageQueue取消息的next方法中洪碳,IdleHandler相關(guān)代碼如下:

    @UnsupportedAppUsage
    Message next() {
        ...
        int pendingIdleHandlerCount = -1; // -1 only during first iteration
        int nextPollTimeoutMillis = 0;
        for (;;) {
            if (nextPollTimeoutMillis != 0) {
                Binder.flushPendingCommands();
            }

            nativePollOnce(ptr, nextPollTimeoutMillis);

            synchronized (this) {
                ...
                if (msg != null) {
                 ...

                // If first time idle, then get the number of idlers to run.
                // Idle handles only run if the queue is empty or if the first message
                // in the queue (possibly a barrier) is due to be handled in the future.
                if (pendingIdleHandlerCount < 0
                        && (mMessages == null || now < mMessages.when)) {
                   //如果消息隊(duì)列為空或者消息執(zhí)行時(shí)間還未到,則獲取IdleHandler隊(duì)列的大小叼屠,下面需要用到
                    pendingIdleHandlerCount = mIdleHandlers.size();
                }
                if (pendingIdleHandlerCount <= 0) {
                    // No idle handlers to run.  Loop and wait some more.
                   //無需要執(zhí)行的  idle handler瞳腌,則繼續(xù)阻塞
                    mBlocked = true;
                    continue;
                }

                if (mPendingIdleHandlers == null) {
                    mPendingIdleHandlers = new IdleHandler[Math.max(pendingIdleHandlerCount, 4)];
                }
                //將IdleHandler列表轉(zhuǎn)為數(shù)組
                mPendingIdleHandlers = mIdleHandlers.toArray(mPendingIdleHandlers);
            }

            // Run the idle handlers.
            // We only ever reach this code block during the first iteration.
            for (int i = 0; i < pendingIdleHandlerCount; i++) {
                final IdleHandler idler = mPendingIdleHandlers[i];
                mPendingIdleHandlers[i] = null; // release the reference to the handler

                boolean keep = false;
                try {
                    keep = idler.queueIdle();   //開始順序執(zhí)行所有IdleHandler的queueIdle方法
                } catch (Throwable t) {
                    Log.wtf(TAG, "IdleHandler threw exception", t);
                }

                if (!keep) {   //如果發(fā)現(xiàn)有queueIdle()方法返回false,則線程安全地刪除這個(gè)idlehandler不再執(zhí)行queueIdle
                    synchronized (this) {
                        mIdleHandlers.remove(idler);
                    }
                }
            }
            ...
        }
    }
多任務(wù)延遲初始化實(shí)戰(zhàn):

我們根據(jù)queueIdle返回true時(shí)可以執(zhí)行多次的特點(diǎn)镜雨,可以實(shí)現(xiàn)一個(gè)任務(wù)列表嫂侍,然后從這個(gè)任務(wù)列表中取任務(wù)執(zhí)行。

public class TaskDispatcher {

    private Queue<Runnable> delayTasks = new LinkedList<>();

    private MessageQueue.IdleHandler idleHandler = () -> {
        if (delayTasks.size() > 0) {
            Runnable task = delayTasks.poll();
            if (task != null) {
                task.run();
            }
        }
        return !delayTasks.isEmpty();  //只要task任務(wù)不為空荚坞,就繼續(xù)執(zhí)行初始化
    };

    public TaskDispatcher addTask(Runnable task) {
        delayTasks.add(task);
        return this;
    }

    public void start() {
        Looper.myQueue().addIdleHandler(idleHandler);
    }
}

創(chuàng)建一個(gè)ARouter初始化和Webview初始的task

public class WebviewInitTask implements Runnable {

    @Override
    public void run() {
        Log.i("minfo", "初始化Okhttp");
    }
}

public class WebviewInitTask implements Runnable {

    @Override
    public void run() {
        Log.i("minfo", "初始化Webview");
    }
}

界面顯示后進(jìn)行調(diào)用:

override fun onCreate(savedInstanceState: Bundle?) {
        setTheme(R.style.AppTheme)
        super.onCreate(savedInstanceState)
        
            TaskDispatcher()
            .addTask(ARouterInitTask())
            .addTask(WebviewInitTask())
            .start()
    }

打印執(zhí)行結(jié)果


關(guān)于IdleHandler問題

Q:IdleHandler 有什么用挑宠?

IdleHandler 是 Handler 提供的一種在消息隊(duì)列空閑時(shí),執(zhí)行任務(wù)的時(shí)機(jī)颓影;
當(dāng) MessageQueue 當(dāng)前沒有立即需要處理的消息時(shí)各淀,會(huì)執(zhí)行 IdleHandler;

Q:MessageQueue 提供了 add/remove IdleHandler 的方法诡挂,是否需要成對(duì)使用碎浇?

不是必須;
IdleHandler.queueIdle() 的返回值璃俗,可以移除加入 MessageQueue 的 IdleHandler奴璃;
Q:當(dāng) mIdleHanders 一直不為空時(shí),為什么不會(huì)進(jìn)入死循環(huán)旧找?

只有在 pendingIdleHandlerCount 為 -1 時(shí)溺健,才會(huì)嘗試執(zhí)行 mIdleHander;
pendingIdlehanderCount 在 next() 中初始時(shí)為 -1,執(zhí)行一遍后被置為 0鞭缭,所以不會(huì)重復(fù)執(zhí)行剖膳;
Q:是否可以將一些不重要的啟動(dòng)服務(wù),搬移到 IdleHandler 中去處理岭辣?

不建議吱晒;
IdleHandler 的處理時(shí)機(jī)不可控,如果 MessageQueue 一直有待處理的消息沦童,那么 IdleHander 的執(zhí)行時(shí)機(jī)會(huì)很靠后仑濒;
Q:IdleHandler 的 queueIdle() 運(yùn)行在那個(gè)線程?

陷進(jìn)問題偷遗,queueIdle() 運(yùn)行的線程墩瞳,只和當(dāng)前 MessageQueue 的 Looper 所在的線程有關(guān);
子線程一樣可以構(gòu)造 Looper氏豌,并添加 IdleHandler喉酌;

參考:
https://juejin.cn/post/7055564669540368392

Github代碼地址:

https://github.com/running-libo/PerformanceOpt

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市泵喘,隨后出現(xiàn)的幾起案子泪电,更是在濱河造成了極大的恐慌,老刑警劉巖纪铺,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件相速,死亡現(xiàn)場離奇詭異,居然都是意外死亡鲜锚,警方通過查閱死者的電腦和手機(jī)突诬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來烹棉,“玉大人攒霹,你說我怎么就攤上這事怯疤〗矗” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵集峦,是天一觀的道長伏社。 經(jīng)常有香客問我,道長塔淤,這世上最難降的妖魔是什么摘昌? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任高蜂,我火速辦了婚禮,結(jié)果婚禮上备恤,老公的妹妹穿的比我還像新娘稿饰。我一直安慰自己,他們只是感情好喉镰,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著侣姆,像睡著了一般。 火紅的嫁衣襯著肌膚如雪捺宗。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天蚜厉,我揣著相機(jī)與錄音,去河邊找鬼弯囊。 笑死痰哨,一個(gè)胖子當(dāng)著我的面吹牛匾嘱,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播霎烙,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼撬讽,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼!你這毒婦竟也來了悬垃?” 一聲冷哼從身側(cè)響起游昼,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎尝蠕,沒想到半個(gè)月后烘豌,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡看彼,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年廊佩,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片靖榕。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡标锄,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出茁计,到底是詐尸還是另有隱情料皇,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布,位于F島的核電站践剂,受9級(jí)特大地震影響毒返,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜舷手,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一拧簸、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧男窟,春花似錦盆赤、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至汗捡,卻和暖如春淑际,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背扇住。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來泰國打工春缕, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人艘蹋。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓锄贼,卻偏偏與公主長得像,于是被迫代替她去往敵國和親女阀。 傳聞我的和親對(duì)象是個(gè)殘疾皇子宅荤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容