轉(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