Handler.post解惑

android美隊.png

今日修改bug拿到一份代碼蔚袍,其中網(wǎng)絡(luò)請求方面由于功能很簡單乡范,就沒有使用網(wǎng)絡(luò)請求框架,ok啤咽,那按照我的想法晋辆,開啟一個線程拿個數(shù)據(jù),拿完發(fā)送handler更新UI即可了闰蚕,但是代碼中并不是這么寫的栈拖,而是通過mNetHandler.post一個包含網(wǎng)絡(luò)請求的runnable。mNetHandler的來源是這樣的没陡。

        HandlerThread handlerThread = new HandlerThread("NET");
        handlerThread.start();
        mNetHandler = new Handler(handlerThread.getLooper());

而當(dāng)我詢問原作者原因時涩哟,解釋是用mNetHandler來管理這些Runnable,在view結(jié)束時盼玄,

mNetHandler.removeCallback

移除這些runnable贴彼,解決掉在view,或者說activity中開啟線程埃儿,而當(dāng)view或者activity結(jié)束時 線程仍然存活的問題器仗。

但是目前存在一個問題,removeCallback并不是立即停止該線程,而是移除掉還未執(zhí)行的callback精钮,正在執(zhí)行的是無法立即結(jié)束的威鹿。此問題我寫了demo,證實的確是正在運行的程序無論通過
mNetHandler.getLooper().quit();
handlerThread.quitSafely();
都無法停止該線程轨香,我們來看一下我的demo
功能很簡單忽你,activity1點擊button進(jìn)activity2,activity2一進(jìn)去就開啟線程干事情臂容,然后點擊finsh按鈕科雳,結(jié)束當(dāng)前activity,回到activity1脓杉,無論哪種模式糟秘,activity開的那個線程都能存活。

  • Activity 1的代碼 沒什么東西 就是跳轉(zhuǎn)
public class MainActivity extends AppCompatActivity {

    private Button button;
    private TextView textView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView = (TextView) findViewById(R.id.my_text);
        button = (Button) findViewById(R.id.button);
        textView.setText("第一個界面");
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Log.i("iii"," 跳轉(zhuǎn)去activity2");
                Intent intent = new Intent(MainActivity.this,SecondActivity.class);
                startActivity(intent);
            }
        });
    }
}
  • Activity2的代碼
/**
 * Created by xiamin on 11/19/16.
 */
public class SecondActivity extends AppCompatActivity {
    private Button button;
    private TextView textView;
    private Handler mNetHandler;
    HandlerThread handlerThread;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Log.i("iii","進(jìn)入activity2");
        textView = (TextView) findViewById(R.id.my_text);
        button = (Button) findViewById(R.id.button);
        textView.setText("第2個界面");
        button.setText("finish()");
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                finish();
            }
        });

        handlerThread = new HandlerThread("NET");
        handlerThread.start();
        mNetHandler = new Handler(handlerThread.getLooper());
        mNetHandler.post(mRunnable);
    }

    private static int count = 0;
    private Runnable mRunnable = new Runnable() {
        @Override
        public void run() {
            while (true) {
                Log.i("iii"," count = " + count++);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    };

    @Override
    protected void onDestroy() {
        Log.i("iii","activity2 onDestroy() ");
        mNetHandler.removeCallbacksAndMessages(null);
        mNetHandler.getLooper().quit();
        handlerThread.quitSafely();
        super.onDestroy();
    }
}

打印為球散,我們可以見到runnable里在繼續(xù)打印

11-19 17:05:15.930 6280-6280/? I/iii: 跳轉(zhuǎn)去activity2
11-19 17:05:15.960 6280-6280/? I/iii: 進(jìn)入activity2
11-19 17:05:15.960 6280-6321/? I/iii: count = 0
11-19 17:05:16.960 6280-6321/? I/iii: count = 1
11-19 17:05:17.960 6280-6321/? I/iii: count = 2
11-19 17:05:18.960 6280-6321/? I/iii: count = 3
11-19 17:05:19.250 6280-6280/? I/iii: activity2 onDestroy()
11-19 17:05:19.960 6280-6321/? I/iii: count = 4
11-19 17:05:20.960 6280-6321/? I/iii: count = 5
11-19 17:05:21.960 6280-6321/? I/iii: count = 6
11-19 17:05:22.960 6280-6321/? I/iii: count = 7

可見尿赚,使用handlerThread只充當(dāng)了一個線程的單行執(zhí)行器的作用,并沒有能夠起到所謂的控制runnable執(zhí)行的作用沛婴。事實上吼畏,什么都不能控制runnable執(zhí)行的督赤。

  • 這個事情提醒我們幾點:

1.activity結(jié)束后線程還能存活嘁灯,所以我們要記得結(jié)束它或者知道要管理好他們

2.使用handlerThread.post runnable 只是讓handlerThread充當(dāng)了線程的順序執(zhí)行器

  • handler回顧

Android程序員都知道不能在UI線程執(zhí)行耗時的操作,Android引入handler就是為了解決這個問題躲舌,當(dāng)然實現(xiàn)異步更新UI不僅僅只有這一種方法丑婿,還有AsyncTask也可以實現(xiàn)。

Android有一個 Handler類没卸,使用該類可以對運行在不同線程中的多個任務(wù)進(jìn)行排隊羹奉,并使用Message和Runnable對象安排這些任務(wù)。在javadoc中约计,對Handler是這樣解釋的:Handler可以發(fā)送和處理消息對象或Runnable對象诀拭,這些消息對象和Runnable對象與一個線程相關(guān)聯(lián)。每個Handler的實例都關(guān)聯(lián)了一個線程和線程的消息隊列煤蚌。當(dāng)創(chuàng)建了一個Handler對象時耕挨,一個線程或消息隊列同時也被創(chuàng)建,該Handler對象將發(fā)送和處理這些消息或Runnable對象尉桩。

a筒占、如果new一個無參構(gòu)造函數(shù)的Handler對象,那么這個Handler將自動與當(dāng)前運行線程相關(guān)聯(lián)蜘犁,也就是說這個Handler將與當(dāng)前運行的線程使用同一個消息隊列翰苫,并且可以處理該隊列中的消息。
我做過這樣一個實驗,在主用戶界面中創(chuàng)建一個帶有無參構(gòu)造函數(shù)的Handler對象奏窑,該Handler對象向消息隊列推送一個Runnable對象导披,在Runnable對象的run函數(shù)中打印當(dāng)前線程Id,主用戶界面線程ID和Runnable線程ID均為1埃唯。

b盛卡、如果new一個帶參構(gòu)造函數(shù)的Handler對象,那么這個Handler對象將與參數(shù)所表示的Looper相關(guān)聯(lián)筑凫。注意:此時線程類應(yīng)該是一個特殊類HandlerThread類滑沧,一個Looper類的Thread類,它繼承自Thread類巍实。

c滓技、如果需要Handler對象去處理消息,那么就要重載Handler類的handleMessage函數(shù)棚潦。


謝謝大家閱讀令漂,如有幫助,來個喜歡或者關(guān)注吧丸边!


本文作者:Anderson/Jerey_Jobs

簡書地址:[Anderson大碼渣][1]

github地址:[Jerey_Jobs][2]
[1]: http://www.reibang.com/users/016a5ba708a0/latest_articles
[2]: https://github.com/Jerey-Jobs

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末叠必,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子妹窖,更是在濱河造成了極大的恐慌纬朝,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件骄呼,死亡現(xiàn)場離奇詭異共苛,居然都是意外死亡,警方通過查閱死者的電腦和手機蜓萄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評論 3 394
  • 文/潘曉璐 我一進(jìn)店門隅茎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人嫉沽,你說我怎么就攤上這事辟犀。” “怎么了绸硕?”我有些...
    開封第一講書人閱讀 164,057評論 0 354
  • 文/不壞的土叔 我叫張陵堂竟,是天一觀的道長。 經(jīng)常有香客問我臣咖,道長跃捣,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,509評論 1 293
  • 正文 為了忘掉前任夺蛇,我火速辦了婚禮疚漆,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己娶聘,他們只是感情好闻镶,可當(dāng)我...
    茶點故事閱讀 67,562評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著丸升,像睡著了一般铆农。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上狡耻,一...
    開封第一講書人閱讀 51,443評論 1 302
  • 那天墩剖,我揣著相機與錄音,去河邊找鬼夷狰。 笑死岭皂,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的沼头。 我是一名探鬼主播爷绘,決...
    沈念sama閱讀 40,251評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼进倍!你這毒婦竟也來了土至?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評論 0 276
  • 序言:老撾萬榮一對情侶失蹤猾昆,失蹤者是張志新(化名)和其女友劉穎陶因,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體毡庆,經(jīng)...
    沈念sama閱讀 45,561評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡坑赡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,779評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了么抗。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,902評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡亚铁,死狀恐怖蝇刀,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情徘溢,我是刑警寧澤吞琐,帶...
    沈念sama閱讀 35,621評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站然爆,受9級特大地震影響站粟,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜曾雕,卻給世界環(huán)境...
    茶點故事閱讀 41,220評論 3 328
  • 文/蒙蒙 一奴烙、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦切诀、人聲如沸揩环。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽丰滑。三九已至,卻和暖如春倒庵,著一層夾襖步出監(jiān)牢的瞬間褒墨,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評論 1 269
  • 我被黑心中介騙來泰國打工擎宝, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留貌亭,地道東北人。 一個月前我還...
    沈念sama閱讀 48,025評論 2 370
  • 正文 我出身青樓认臊,卻偏偏與公主長得像圃庭,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子失晴,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,843評論 2 354

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