Android日記之消息機(jī)制(2)

前言

上一篇Android日記之消息機(jī)制(1)分析了Handler的用法和實(shí)現(xiàn)原理藏畅,這一篇主要講講ThreadLocal和HandlerThread,這兩個(gè)也是在面試中經(jīng)常被問到的東西,我們就先從ThreadLocal開始講起苹享。

ThreadLocal(基于jdk1.8)

ThreadLocal基本使用

它是一個(gè)線程內(nèi)部的數(shù)據(jù)存儲(chǔ)類逢并,就是線程局部變量,通過它就可以在指定的線程中去存儲(chǔ)數(shù)據(jù)了跺嗽,當(dāng)數(shù)據(jù)存儲(chǔ)完畢后战授,也只有在指定的線程中才可以獲取到存儲(chǔ)的數(shù)據(jù),而其他的線程是無法獲取到這些數(shù)據(jù)的桨嫁。在一般的日常開發(fā)中用的比較少植兰,但是在某些場景下,也可以使用ThreadLocal實(shí)現(xiàn)一些看起來很復(fù)雜的功能璃吧。一般來說楣导,當(dāng)某些數(shù)據(jù)是以線程為作用域并且不同線程具有不同的數(shù)據(jù)副本的時(shí)候,就可以考慮采用 ThreadLocal畜挨。

老樣子筒繁,我們看下基本的使用方法:

//ThreadLocal使用方法
package com.ju.threadlocaldemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;

public class MainActivity extends AppCompatActivity {


    private ThreadLocal<Boolean> threadLocal = new ThreadLocal<>();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        threadLocal.set(true);
        Log.d("ThreadLocal","[main]="+ threadLocal.get());

        new Thread("Thread1"){
            @Override
            public void run() {
                threadLocal.set(false);
                Log.d("ThreadLocal","[Thread1]="+ threadLocal.get());
            }
        }.start();

        new Thread("Thread2"){
            @Override 
            public void run() {
                Log.d("ThreadLocal","[Thread2]="+ threadLocal.get());
            }
        }.start();
    }
}

首先實(shí)例化這個(gè)ThreadLocal,然后傳入你要傳遞的值的泛型朦促,這里為了測試我就傳入Boolen類型膝晾,接下來,你就可以通過set()get()去獲得對應(yīng)的值务冕。這里我們做了一個(gè)測試血当,分別創(chuàng)建兩個(gè)子線程,然后包括主線程在內(nèi)每一個(gè)ThreadLocal都set()get()對應(yīng)的值看看禀忆,結(jié)果會(huì)是怎么樣臊旭。

測試結(jié)果

從Logcat日記中我們可以看出,雖然不同線程訪問的都是同一個(gè)ThreadLocal對象箩退,但是它們獲取到的值卻是不一樣的离熏,這就是ThreadLocal的特別之處了,為什么會(huì)這樣呢戴涝?接下來我們主要分析set()get()的源碼來更深刻的去理解滋戳。

set()源碼分析

public void set(T value) {
    //獲取當(dāng)前線程
    Thread t = Thread.currentThread();
    //獲取當(dāng)前線程的map
    ThreadLocalMap map = getMap(t);
    //這里會(huì)判斷是否之前有沒有調(diào)用過set和get方法,如果沒有啥刻,就為當(dāng)前線程創(chuàng)建一個(gè)ThreadLocalMap奸鸯,
    
    if (map != null)
        map.set(this, value);
    else
    //key為當(dāng)前ThreadLocalMap對象
        createMap(t, value);
}

ThreadLocalMap getMap(Thread t) {
    return t.threadLocals;
}

void createMap(Thread t, T firstValue) {
    t.threadLocals = new ThreadLocalMap(this, firstValue);
}

這里我們看到了ThreadLocalMap這個(gè)類,它是ThreadLocal的里的一個(gè)內(nèi)部類可帽,是用基于線性探測法的散列表實(shí)現(xiàn)的娄涩。set()方法很簡單,看源碼就可以熟悉映跟,蓄拣。

get()源碼分析

public T get() {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    // 這里判斷是否調(diào)用過set或get方法時(shí)扬虚,對象值已經(jīng)設(shè)置過,就返回上一次的值
    if (map != null) {
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    //沒有的話球恤,就自動(dòng)設(shè)置為初始值
    return setInitialValue();
}

private T setInitialValue() {
    T value = initialValue();
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
    return value;
}


//通彻缄牵可以自己去重寫,來設(shè)置初始值
protected T initialValue() {
    return null;
}

代碼也很簡單咽斧,首先判斷是否存在這個(gè)值路鹰,如果有直接返回就好了,如果沒有的話收厨,就通過setInitialValue()這個(gè)方法來穿件一個(gè)初始值進(jìn)行返回晋柱,也可以通過initialValue()這個(gè)來設(shè)置初始值。

HandlerThread

它繼承了一個(gè)Thread诵叁,是一種可以使用Handler的Thread雁竞,是屬于Android多線程Android應(yīng)用開發(fā)的很經(jīng)常使用的類,在IntentService的源碼也有用到HandlerThread拧额,它的實(shí)現(xiàn)也很簡單碑诉,就是在run()方法里通過Looper.prepare()來創(chuàng)建消息隊(duì)列,并通過Looper.loop()來開啟消息循環(huán)侥锦,這樣就可以運(yùn)行可以在HandlerThread里創(chuàng)建Handler了进栽。它的本質(zhì)其實(shí)就是Handler+Thread。官方介紹是這樣的:HandlerThread是Android API提供的一個(gè)方便恭垦、便捷的類快毛,使用它我們可以快速的創(chuàng)建一個(gè)帶有Looper的線程。Looper可以用來創(chuàng)建Handler實(shí)例番挺。

HandlerThread的基本使用

package com.ju.handlerthreaddemo;

import android.os.Bundle;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;

public class MainActivity extends AppCompatActivity {


    private HandlerThread handlerThread = new HandlerThread("HandlerThread");
    private Handler handler;
    private Handler mainHandler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            //這里更新UI
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //開啟線程
        handlerThread.start();
        //創(chuàng)建Handler與handlerThread綁定
        handler = new Handler(handlerThread.getLooper()){
            @Override
            public void handleMessage(Message msg) {
                //這里進(jìn)行耗時(shí)任務(wù)的處理唠帝,處理完畢可以發(fā)送給主Handler進(jìn)行相應(yīng)的操作。
                //也可以通過繼承HandlerThread來處理
                mainHandler.sendMessage(msg);
            }
        };
    }

    //銷毀
    @Override
    protected void onDestroy() {
        super.onDestroy();
        handlerThread.quit();
    }
}

首先先創(chuàng)建HandlerThread的實(shí)例玄柏,然后創(chuàng)建一個(gè)Handler與HandlerThread進(jìn)行綁定通過getLooper()來獲取Looper襟衰,然后一定要使用start()方法開啟線程,這里注意的是handler是運(yùn)行在子線程的粪摘,所以是不能在這里面進(jìn)行更新UI的操作瀑晒,這時(shí)候你就可以在這個(gè)handler里面進(jìn)行耗時(shí)的操作,然后吧結(jié)果可以通過sendMessage()方法發(fā)送給處于UI線程的Handler進(jìn)行更新UI的操作就可以了徘意。最后要記得要銷毀這個(gè)HandlerThread才行苔悦。

HandlerThread的run()源碼解析

因?yàn)镠andlerThread本質(zhì)上其實(shí)就是繼承了Thread,所以我們就就是看Thread最重要的run()方法實(shí)現(xiàn)映砖。

@Override
public void run() {
    mTid = Process.myTid();
    Looper.prepare();
    synchronized (this) {
        mLooper = Looper.myLooper();
        notifyAll();
    }
    Process.setThreadPriority(mPriority);
    onLooperPrepared();
    Looper.loop();
    mTid = -1;
}

從實(shí)現(xiàn)來看间坐,一般的Thread的run()只執(zhí)行一個(gè)耗時(shí)的任務(wù)灾挨,HandlerThread在內(nèi)部創(chuàng)建了一個(gè)消息隊(duì)列邑退,外界需要通過Handler的消息方式通知HandlerThread要執(zhí)行哪一個(gè)具體的任務(wù)竹宋,這里注意一下,當(dāng)不需要使用HandlerThread時(shí)地技,就要通過quit()或者quitSafely方法來終止線程的執(zhí)行蜈七。

參考

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市莫矗,隨后出現(xiàn)的幾起案子飒硅,更是在濱河造成了極大的恐慌,老刑警劉巖作谚,帶你破解...
    沈念sama閱讀 211,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件三娩,死亡現(xiàn)場離奇詭異,居然都是意外死亡妹懒,警方通過查閱死者的電腦和手機(jī)雀监,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,347評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來眨唬,“玉大人会前,你說我怎么就攤上這事∝腋停” “怎么了瓦宜?”我有些...
    開封第一講書人閱讀 157,435評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長岭妖。 經(jīng)常有香客問我临庇,道長,這世上最難降的妖魔是什么昵慌? 我笑而不...
    開封第一講書人閱讀 56,509評(píng)論 1 284
  • 正文 為了忘掉前任苔巨,我火速辦了婚禮,結(jié)果婚禮上废离,老公的妹妹穿的比我還像新娘侄泽。我一直安慰自己,他們只是感情好蜻韭,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,611評(píng)論 6 386
  • 文/花漫 我一把揭開白布悼尾。 她就那樣靜靜地躺著,像睡著了一般肖方。 火紅的嫁衣襯著肌膚如雪闺魏。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,837評(píng)論 1 290
  • 那天俯画,我揣著相機(jī)與錄音析桥,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛泡仗,可吹牛的內(nèi)容都是我干的埋虹。 我是一名探鬼主播,決...
    沈念sama閱讀 38,987評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼娩怎,長吁一口氣:“原來是場噩夢啊……” “哼搔课!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起截亦,我...
    開封第一講書人閱讀 37,730評(píng)論 0 267
  • 序言:老撾萬榮一對情侶失蹤爬泥,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后崩瓤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體袍啡,經(jīng)...
    沈念sama閱讀 44,194評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,525評(píng)論 2 327
  • 正文 我和宋清朗相戀三年却桶,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了葬馋。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,664評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡肾扰,死狀恐怖畴嘶,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情集晚,我是刑警寧澤窗悯,帶...
    沈念sama閱讀 34,334評(píng)論 4 330
  • 正文 年R本政府宣布,位于F島的核電站偷拔,受9級(jí)特大地震影響蒋院,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜莲绰,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,944評(píng)論 3 313
  • 文/蒙蒙 一欺旧、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蛤签,春花似錦辞友、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,764評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至戳晌,卻和暖如春鲫尊,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背沦偎。 一陣腳步聲響...
    開封第一講書人閱讀 31,997評(píng)論 1 266
  • 我被黑心中介騙來泰國打工疫向, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留咳蔚,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,389評(píng)論 2 360
  • 正文 我出身青樓搔驼,卻偏偏與公主長得像谈火,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子匙奴,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,554評(píng)論 2 349

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