Android多線程:IntentService

前言

Service作為Android的四大組件之一萄涯,實現(xiàn)了在后臺執(zhí)行不需要界面的任務,例如音樂的播放等唆鸡,若要實現(xiàn)一個在后臺執(zhí)行的耗時任務呢涝影?例如離線下載。于是争占,Android 實現(xiàn)了 IntentService 燃逻,通過開啟一個子線程在后臺執(zhí)行耗時任務序目。

1. 使用步驟

1)新建IntentService子類MyIntentService
2)重寫 onHandleIntent() 方法,根據(jù)傳入的不同Intent處理不同的任務
3)在 AndroidManifest.xml 中注冊服務MyIntentService
4)需要發(fā)起任務請求的地方唆樊,新建 Intent 對象宛琅,使用 Bundle 保存?zhèn)魅氲臄?shù)據(jù)
5)調(diào)用 ActivitystartService(intent) 發(fā)起一次任務請求

MyIntentService.java

//step1: 新建IntentService子類MyIntentService
public class MyIntentService extends IntentService {

    private String taskName;

    //step2: 調(diào)用父類構(gòu)造方法,傳入線程名字
    public MyIntentService() {
        super("myIntentService");
    }

    @Override
    public void onCreate() {
        super.onCreate();
        Log.i("myIntentService----", "onCreate");
    }

    //默認實現(xiàn)逗旁,將傳入的Intent請求依次加入到消息隊列中
    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        Log.i("myIntentService----", "onStartCommand");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.i("myIntentService----", "onDestroy");
    }
}

AndroidManifest.xml

<service
       android:name=".MyIntentService"
       tools:ignore="ExportedService"/>

發(fā)起任務請求的方法:

    //在需要發(fā)起任務請求的地方調(diào)用此方法
    private void startIntentService() {

        //step5: 新建Intent傳入任務處理請求嘿辟,指向MyIntentService
        Intent task1 = new Intent(MainActivity.this, MyIntentService.class);
        Bundle bundle1 = new Bundle();
        bundle1.putString("taskName", "task1");
        task1.putExtras(bundle1);

        Intent task2 = new Intent(MainActivity.this, MyIntentService.class);
        Bundle bundle2 = new Bundle();
        bundle2.putString("taskName", "task2");
        task2.putExtras(bundle2);

        //step6: 啟動服務,發(fā)起一次任務請求 (多次調(diào)用startService(intent) 則發(fā)起多次請求)
        startService(task1);
        startService(task2);
        //再次發(fā)起task1的任務請求
        startService(task1);
    }
  • bug記錄
    在AndoridManifest.xml文件中注冊MyIntentService時片效,為Service添加了屬性name红伦,并在代碼中用包名的方式調(diào)用報錯:
    Service Intent must be explicit: Intent
    原因:Android 5.0(Lollipop) 之后規(guī)定不能以包名的方式定義Service的Intent,而應用顯示聲明 new Intent(xxxActivity.this, xxxService.class);

2. 源碼分析

1)IntentService的初始化

    //step2: 調(diào)用父類構(gòu)造方法淀衣,傳入線程名字
    public MyIntentService() {
        super("myIntentService");
    }

    //調(diào)用了此構(gòu)造方法昙读,賦值線程名稱mName
    public IntentService(String name) {
        super();
        mName = name;
    }
    
    //IntentService的onCreate()中
    @Override
    public void onCreate() {
        super.onCreate();
        //下面即是開啟一個HandlerThread線程的常用步驟,詳細可參考之前文章中有關HandlerThread的解析
        //創(chuàng)建一個新線程HandlerThread對象膨桥,傳入線程名稱mName
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        //啟動線程
        thread.start();
        //獲取HandlerThread自動創(chuàng)建的Looper
        mServiceLooper = thread.getLooper();
        //關聯(lián)HandlerThread和IntentService的Handler對象
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }

2)onHandleIntent(Intent intent)

    //step3: 重寫onHandleIntent蛮浑,判斷Intent類型,依次處理不同的任務
    @Override
    protected void onHandleIntent(Intent intent) {
        ...
    }

    //IntentService中只嚣,創(chuàng)建ServiceHandler類沮稚,繼承自Hanlder
    private final class ServiceHandler extends Handler {
        //構(gòu)造方法,傳入ServiceHandler所在線程的Looper對象
        public ServiceHandler(Looper looper) {
            super(looper);
        }
        //復寫handleMessage册舞,處理任務請求
        @Override
        public void handleMessage(Message msg) {
            //進入子類MyIntentService中重寫的構(gòu)造方法onHandleIntent(intent)蕴掏,這個方法的執(zhí)行是在工作線程中
            onHandleIntent((Intent)msg.obj);
            //調(diào)用Service的stopSelf() 自動停止開啟的IntentService服務,標記為startId
            stopSelf(msg.arg1);
        }
    }
    //子類必須實現(xiàn)的構(gòu)造方法调鲸,工作線程執(zhí)行盛杰,根據(jù)Intent
    @WorkerThread
    protected abstract void onHandleIntent(@Nullable Intent intent);

3)onStartCommand(intent, flags, startId)

    //將傳入的Intent請求依次加入到消息隊列中
    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        return super.onStartCommand(intent, flags, startId);
    }
    
    //IntentService中onStartCommand
    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        //調(diào)用onStart(),傳入發(fā)起的任務請求Intent和該次請求的id值startId
        onStart(intent, startId);//——→跳轉(zhuǎn)至onStart()方法
        return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
    }

    //onStart()方法
    @Override
    public void onStart(@Nullable Intent intent, int startId) {
        //新建Message對象藐石,存儲intent和startId
        Message msg = mServiceHandler.obtainMessage();
        msg.arg1 = startId;
        msg.obj = intent;
        //ServiceHandler發(fā)送這條消息Message到消息隊列MessageQueue中即供,等待對應的Handler處理消息任務
        mServiceHandler.sendMessage(msg);
    }

4)onDestroy()

    @Override
    public void onDestroy() {
        //IntentService結(jié)束即MessageQueue內(nèi)部的Message處理完成時,
        //自動退出當前線程所持有的Looper于微,Looper自動退出所持有的MessageQueue
        mServiceLooper.quit();
    }

5)onBind(intent)

    @Override
    @Nullable
    public IBinder onBind(Intent intent) {
        return null;
        //返回為空
    }

啟動IntentService時募狂,若采用bindService()方式啟動,IntentService的生命周期為:
onCrtate() → onBind() → onUnBind() → onDestroy()
若采用startService()方式啟動角雷,IntentService的生命周期為:
onCrtate() → onStartCommand() → onDestroy()
從生命周期來看,bindService()不會執(zhí)行onStartCommand()性穿,而IntentService在onStartCommand()方法中實現(xiàn)消息的創(chuàng)建和發(fā)送入隊操作勺三,所以IntentService只能通過startService()的方式啟動才能實現(xiàn)HandlerThread線程發(fā)送消息并在后臺Service中執(zhí)行耗時任務。

3. 應用場景

線程任務需要 按序 且在 后臺執(zhí)行 需曾,所有任務都在同一個Thread的Looper中執(zhí)行

  • 離線下載

思考

  • 為什么IntentService是 Service + HandlerThread + Handler 的結(jié)合吗坚?

1)IntentService 繼承自 Service祈远;
2)在 IntentService 的 onCreate() 方法中,創(chuàng)建 HandlerThread 對象商源,開啟子線程執(zhí)行耗時任務车份;
3)創(chuàng)建內(nèi)部類 ServiceHandler 繼承自 Handler ,實現(xiàn)消息的發(fā)送和處理牡彻。

  • 為什么IntentService會在執(zhí)行完所有任務請求后自動停止Service?

內(nèi)部類ServiceHandler的handleMessage(msg)方法中扫沼,調(diào)用了父類Service方法stopSelf(),停止了IntentService庄吼。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末缎除,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子总寻,更是在濱河造成了極大的恐慌器罐,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件渐行,死亡現(xiàn)場離奇詭異轰坊,居然都是意外死亡,警方通過查閱死者的電腦和手機祟印,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進店門肴沫,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人旁理,你說我怎么就攤上這事樊零。” “怎么了孽文?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵驻襟,是天一觀的道長。 經(jīng)常有香客問我芋哭,道長沉衣,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任减牺,我火速辦了婚禮豌习,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘拔疚。我一直安慰自己肥隆,他們只是感情好,可當我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布稚失。 她就那樣靜靜地躺著栋艳,像睡著了一般。 火紅的嫁衣襯著肌膚如雪句各。 梳的紋絲不亂的頭發(fā)上吸占,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天晴叨,我揣著相機與錄音,去河邊找鬼矾屯。 笑死兼蕊,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的件蚕。 我是一名探鬼主播孙技,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼骤坐!你這毒婦竟也來了绪杏?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤纽绍,失蹤者是張志新(化名)和其女友劉穎蕾久,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體拌夏,經(jīng)...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡僧著,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了障簿。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片盹愚。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖站故,靈堂內(nèi)的尸體忽然破棺而出皆怕,到底是詐尸還是另有隱情,我是刑警寧澤西篓,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布愈腾,位于F島的核電站,受9級特大地震影響岂津,放射性物質(zhì)發(fā)生泄漏虱黄。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一吮成、第九天 我趴在偏房一處隱蔽的房頂上張望橱乱。 院中可真熱鬧,春花似錦粱甫、人聲如沸泳叠。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽析二。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間叶摄,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工安拟, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蛤吓,地道東北人。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓糠赦,卻偏偏與公主長得像会傲,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子拙泽,可洞房花燭夜當晚...
    茶點故事閱讀 44,927評論 2 355

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