Android IntentService完全解析 當Service遇到Handler

轉(zhuǎn)載請標明出處: http://blog.csdn.net/lmj623565791/article/details/47143563肄渗; 本文出自:【張鴻洋的博客】

一 概述
大家都清楚,在Android的開發(fā)中咬最,凡是遇到耗時的操作盡可能的會交給Service去做翎嫡,比如我們上傳多張圖,上傳的過程用戶可能將應用置于后臺永乌,然后干別的去了惑申,我們的Activity就很可能會被殺死,所以可以考慮將上傳操作交給Service去做翅雏,如果擔心Service被殺圈驼,還能通過設置startForeground(int, Notification)
方法提升其優(yōu)先級。
那么望几,在Service里面我們肯定不能直接進行耗時操作绩脆,一般都需要去開啟子線程去做一些事情,自己去管理Service的生命周期以及子線程并非是個優(yōu)雅的做法橄抹;好在Android給我們提供了一個類靴迫,叫做IntentService
,我們看下注釋楼誓。
IntentService is a base class for {@link Service}s that handle asynchronous requests (expressed as {@link Intent}s) on demand. Clients send requests through {@link android.content.Context#startService(Intent)} calls; the service is started as needed, handles each Intent in turn using a worker thread, and stops itself when it runs out of work.

意思說IntentService是一個基于Service的一個類玉锌,用來處理異步的請求。你可以通過startService(Intent)來提交請求疟羹,該Service會在需要的時候創(chuàng)建主守,當完成所有的任務以后自己關閉禀倔,且請求是在工作線程處理的。
這么說参淫,我們使用了IntentService最起碼有兩個好處救湖,一方面不需要自己去new Thread了;另一方面不需要考慮在什么時候關閉該Service了涎才。
好了鞋既,那么接下來我們就來看一個完整的例子。
二 IntentService的使用
我們就來演示一個多個圖片上傳的案例憔维,當然我們會模擬上傳的耗時涛救,畢竟我們的重心在IntentService的使用和源碼解析上。
首先看下效果圖
效果圖


每當我們點擊一次按鈕业扒,會將一個任務交給后臺的Service去處理检吆,后臺的Service每處理完成一個請求就會反饋給Activity,然后Activity去更新UI程储。當所有的任務完成的時候蹭沛,后臺的Service會退出,不會占據(jù)任何內(nèi)存章鲤。

 package com.zhy.blogcodes.intentservice;

  import android.app.IntentService;
  import android.content.Context;
  import android.content.Intent;
  import android.util.Log;

public class UploadImgService extends IntentService
{
private static final String ACTION_UPLOAD_IMG = "com.zhy.blogcodes.intentservice.action.UPLOAD_IMAGE";
public static final String EXTRA_IMG_PATH = "com.zhy.blogcodes.intentservice.extra.IMG_PATH";

public static void startUploadImg(Context context, String path)
{
    Intent intent = new Intent(context, UploadImgService.class);
    intent.setAction(ACTION_UPLOAD_IMG);
    intent.putExtra(EXTRA_IMG_PATH, path);
    context.startService(intent);
}


public UploadImgService()
{
    super("UploadImgService");
}

@Override
protected void onHandleIntent(Intent intent)
{
    if (intent != null)
    {
        final String action = intent.getAction();
        if (ACTION_UPLOAD_IMG.equals(action))
        {
            final String path = intent.getStringExtra(EXTRA_IMG_PATH);
            handleUploadImg(path);
        }
    }
}

private void handleUploadImg(String path)
{
    try
    {
        //模擬上傳耗時
        Thread.sleep(3000);

        Intent intent = new Intent(IntentServiceActivity.UPLOAD_RESULT);
        intent.putExtra(EXTRA_IMG_PATH, path);
        sendBroadcast(intent);

    } catch (InterruptedException e)
    {
        e.printStackTrace();
    }


}

@Override
public void onCreate()
{
    super.onCreate();
    Log.e("TAG","onCreate");
}

@Override
public void onDestroy()
{
    super.onDestroy();
    Log.e("TAG","onDestroy");
}
}

代碼很短摊灭,主要就是繼承IntentService
,然后復寫onHandleIntent方法败徊,根據(jù)傳入的intent來選擇具體的操作帚呼。startUploadImg
是我寫的一個輔助方法,省的每次都去構建Intent皱蹦,startService了煤杀。

  package com.zhy.blogcodes.intentservice;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
  import android.support.v7.app.AppCompatActivity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.zhy.blogcodes.R;

public class IntentServiceActivity extends AppCompatActivity
{

public static final String UPLOAD_RESULT = "com.zhy.blogcodes.intentservice.UPLOAD_RESULT";

private LinearLayout mLyTaskContainer;

private BroadcastReceiver uploadImgReceiver = new BroadcastReceiver()
{
    @Override
    public void onReceive(Context context, Intent intent)
    {
        if (intent.getAction() == UPLOAD_RESULT)
        {
            String path = intent.getStringExtra(UploadImgService.EXTRA_IMG_PATH);

            handleResult(path);

        }

    }
};

private void handleResult(String path)
{
    TextView tv = (TextView) mLyTaskContainer.findViewWithTag(path);
    tv.setText(path + " upload success ~~~ ");
}


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

    mLyTaskContainer = (LinearLayout) findViewById(R.id.id_ll_taskcontainer);

    registerReceiver();
}

private void registerReceiver()
{
    IntentFilter filter = new IntentFilter();
    filter.addAction(UPLOAD_RESULT);
    registerReceiver(uploadImgReceiver, filter);
}

int i = 0;

public void addTask(View view)
{
    //模擬路徑
    String path = "/sdcard/imgs/" + (++i) + ".png";
    UploadImgService.startUploadImg(this, path);

    TextView tv = new TextView(this);
    mLyTaskContainer.addView(tv);
    tv.setText(path + " is uploading ...");
    tv.setTag(path);
}


@Override
protected void onDestroy()
{
    super.onDestroy();
    unregisterReceiver(uploadImgReceiver);
}
}

Activity中,每當我點擊一次按鈕調(diào)用addTask沪哺,就回模擬創(chuàng)建一個任務沈自,然后交給IntentService去處理。
注意辜妓,當Service的每個任務完成的時候枯途,會發(fā)送一個廣播,我們在Activity的onCreate和onDestroy里面分別注冊和解注冊了廣播籍滴;當收到廣播則更新指定的UI酪夷。
布局文件

      <LinearLayout android:id="@+id/id_ll_taskcontainer"
          xmlns:android="http://schemas.android.com/apk/res/android"
          xmlns:tools="http://schemas.android.com/tools"
          android:layout_width="match_parent"
          android:layout_height="match_parent"
          android:orientation="vertical"
         >


<Button android:layout_width="wrap_content" android:layout_height="wrap_content"
        android:onClick="addTask" android:text="add Task"/>

</LinearLayout>
ok,這樣我們就完成了我們的效果圖的需求异逐;通過上例捶索,大家可以看到我們可以使用IntentService非常方便的處理后臺任務,屏蔽了諸多細節(jié)灰瞻;而Service與Activity通信呢腥例,我們選擇了廣播的方式(當然這里也可以使用LocalBroadcastManager
)。
學會了使用之后酝润,我們再一鼓作氣的看看其內(nèi)部的實現(xiàn)燎竖。
三 IntentService源碼解析
直接看IntentService源碼

  package android.app;

android.content.Intent;
import android.os.Handler;
import android.os.HandlerThread;
import android.os.IBinder;
import android.os.Looper;
import android.os.Message;


public abstract class IntentService extends Service {
private volatile Looper mServiceLooper;
private volatile ServiceHandler mServiceHandler;
private String mName;
private boolean mRedelivery;

private final class ServiceHandler extends Handler {
    public ServiceHandler(Looper looper) {
        super(looper);
    }

    @Override
    public void handleMessage(Message msg) {
        onHandleIntent((Intent)msg.obj);
        stopSelf(msg.arg1);
    }
}


public IntentService(String name) {
    super();
    mName = name;
}


public void setIntentRedelivery(boolean enabled) {
    mRedelivery = enabled;
}

@Override
public void onCreate() {
            super.onCreate();
    HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
    thread.start();

    mServiceLooper = thread.getLooper();
    mServiceHandler = new ServiceHandler(mServiceLooper);
}

@Override
public void onStart(Intent intent, int startId) {
    Message msg = mServiceHandler.obtainMessage();
    msg.arg1 = startId;
    msg.obj = intent;
    mServiceHandler.sendMessage(msg);
}


@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    onStart(intent, startId);
    return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}

@Override
public void onDestroy() {
    mServiceLooper.quit();
}


@Override
public IBinder onBind(Intent intent) {
    return null;
}


protected abstract void onHandleIntent(Intent intent);

}

可以看到它在onCreate里面初始化了一個HandlerThread,關于HandlerThread的使用和源碼 分析參考:Android HandlerThread 完全解析要销,看到這估計已經(jīng)能猜到它的邏輯了:
就是每次調(diào)用onStartCommand的時候构回,通過mServiceHandler發(fā)送一個消息,消息中包含我們的intent疏咐。然后在該mServiceHandler的handleMessage中去回調(diào)onHandleIntent(intent);就可以了纤掸。
那么我們具體看一下源碼,果然是這樣浑塞,onStartCommand中回調(diào)了onStart借跪,onStart中通過mServiceHandler發(fā)送消息到該handler的handleMessage中去。最后handleMessage中回調(diào)onHandleIntent(intent)酌壕。
注意下:回調(diào)完成后回調(diào)用 stopSelf(msg.arg1)掏愁,注意這個msg.arg1是個int值,相當于一個請求的唯一標識卵牍。每發(fā)送一個請求果港,會生成一個唯一的標識,然后將請求放入隊列糊昙,當全部執(zhí)行完成(最后一個請求也就相當于getLastStartId == startId)辛掠,或者當前發(fā)送的標識是最近發(fā)出的那一個(getLastStartId == startId),則會銷毀我們的Service.
如果傳入的是-1則直接銷毀释牺。
那么萝衩,當任務完成銷毀Service回調(diào)onDestory,可以看到在onDestroy中釋放了我們的Looper:mServiceLooper.quit()船侧。
ok~ 如果你的需求可以使用IntentService來做欠气,可以盡可能的使用,設計的還是相當贊的镜撩。當然了预柒,如果你需要考慮并發(fā)等等需求,那么可能需要自己去擴展創(chuàng)建線程池等袁梗。
源碼點擊下載
ok~~
歡迎關注我的微博http://weibo.com/u/3165018720

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末宜鸯,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子遮怜,更是在濱河造成了極大的恐慌淋袖,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,914評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件锯梁,死亡現(xiàn)場離奇詭異即碗,居然都是意外死亡焰情,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評論 2 383
  • 文/潘曉璐 我一進店門剥懒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來内舟,“玉大人,你說我怎么就攤上這事初橘⊙橛危” “怎么了?”我有些...
    開封第一講書人閱讀 156,531評論 0 345
  • 文/不壞的土叔 我叫張陵保檐,是天一觀的道長耕蝉。 經(jīng)常有香客問我,道長夜只,這世上最難降的妖魔是什么垒在? 我笑而不...
    開封第一講書人閱讀 56,309評論 1 282
  • 正文 為了忘掉前任,我火速辦了婚禮盐肃,結果婚禮上爪膊,老公的妹妹穿的比我還像新娘。我一直安慰自己砸王,他們只是感情好推盛,可當我...
    茶點故事閱讀 65,381評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著谦铃,像睡著了一般耘成。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上驹闰,一...
    開封第一講書人閱讀 49,730評論 1 289
  • 那天瘪菌,我揣著相機與錄音,去河邊找鬼嘹朗。 笑死师妙,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的屹培。 我是一名探鬼主播默穴,決...
    沈念sama閱讀 38,882評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼褪秀!你這毒婦竟也來了蓄诽?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,643評論 0 266
  • 序言:老撾萬榮一對情侶失蹤媒吗,失蹤者是張志新(化名)和其女友劉穎仑氛,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,095評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡锯岖,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,448評論 2 325
  • 正文 我和宋清朗相戀三年介袜,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片嚎莉。...
    茶點故事閱讀 38,566評論 1 339
  • 序言:一個原本活蹦亂跳的男人離奇死亡米酬,死狀恐怖沛豌,靈堂內(nèi)的尸體忽然破棺而出趋箩,到底是詐尸還是另有隱情,我是刑警寧澤加派,帶...
    沈念sama閱讀 34,253評論 4 328
  • 正文 年R本政府宣布叫确,位于F島的核電站,受9級特大地震影響芍锦,放射性物質(zhì)發(fā)生泄漏竹勉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,829評論 3 312
  • 文/蒙蒙 一娄琉、第九天 我趴在偏房一處隱蔽的房頂上張望次乓。 院中可真熱鬧,春花似錦孽水、人聲如沸票腰。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,715評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽杏慰。三九已至,卻和暖如春炼鞠,著一層夾襖步出監(jiān)牢的瞬間缘滥,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,945評論 1 264
  • 我被黑心中介騙來泰國打工谒主, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留朝扼,地道東北人。 一個月前我還...
    沈念sama閱讀 46,248評論 2 360
  • 正文 我出身青樓霎肯,卻偏偏與公主長得像擎颖,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子姿现,可洞房花燭夜當晚...
    茶點故事閱讀 43,440評論 2 348

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