1.IntentService是什麼
IntentService是Service的一個派生類,顧名思義是一個Service。我們都知道,對於一些不需要跟隨Activity生命週期的操作,例如上傳、下載等我們在Service裏面進行,但是因爲Service是在UI線程的,所以如果直接在Service裏面進行耗時操作的話,就會導致ANR,所以我們常見的做法是在Service裏面創建子線程進行耗時操作。這樣的做法就會導致Service裏面的線程管理起來比較麻煩,同時還需要創建Looper、Handler實現消息通信,在不使用的時候還需要將Service銷燬。這樣無疑會增加工作量,同時也並不會是比較優雅的做法。
那麼IntentService就是爲了解決這個問題而出現的,在IntentService是一個可以執行異步任務的Service,同時在執行完成異步任務後,會自動銷燬的Service,實現了代碼封裝的同時,避免了資源的浪費。
2.IntentService的實現原理
如下代碼所示,可以看到源碼並不是很多,但是整個設計還是非常優雅的,它裏面主要是通過HandlerThread創建了一個子線程,HandlerThread又自動生成了一個Looper,所以IntentService就可以創建本地線程的Handler,並傳入HandlerThread的Looper,實現消息處理。
因爲每次通過Intent打開Service,都會執行onStartComand操作,它在裏面執行了onStart方法,onStart方法裏面會執行onHandleIntnet這個抽象方法,以供實現類在裏面進行耗時操作,進行完onHandleIntent之後,就會調用stopSelf方法停止該Service,釋放資源。
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(@Nullable Intent intent, int startId) {
Message msg = mServiceHandler.obtainMessage();
msg.arg1 = startId;
msg.obj = intent;
mServiceHandler.sendMessage(msg);
}
@Override
public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
onStart(intent, startId);
return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}
@Override
public void onDestroy() {
mServiceLooper.quit();
}
@Override
@Nullable
public IBinder onBind(Intent intent) {
return null;
}
@WorkerThread
protected abstract void onHandleIntent(@Nullable Intent intent);
}