Android-IntentService源码解析
2019-04-23 本文已影响0人
九号锅炉
在onCreate中创建了一个HandlerThread和Handler.因此和Service处理耗时操作需要另外新建Thread不同,IntentService内置HandlerThread不需要新建线程。同时使用Handler消息机制处理任务意味着是串行处理任务,无法实现并发执行。
@Override
public void onCreate() {
// TODO: It would be nice to have an option to hold a partial wakelock
// during processing, and to have a static startService(Context, Intent)
// method that would launch the service & hand off a wakelock.
super.onCreate();
// 创建HandlerThread, 耗时任务可以在此进行处理
HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
thread.start();
mServiceLooper = thread.getLooper();
mServiceHandler = new ServiceHandler(mServiceLooper);
}
在onStartCommand中调用sendMessage.应用每调用一次startService都会往消息队列中发送一个Message. Message中包含了两个参数:obj是intent,表示这里的Intent 就是启动服务时startService(Intent) 里的 Intent; arg1是startId,就是启动该service生成的id,每次启动生成的id不同。startId可以判断当前启动的service任务是不是消息队列中的最后一个,如果是,立即结束服务。
@Override
public void onStart(@Nullable Intent intent, int startId) {
Message msg = mServiceHandler.obtainMessage();
msg.arg1 = startId;
msg.obj = intent;
mServiceHandler.sendMessage(msg);
}
/**
* You should not override this method for your IntentService. Instead,
* override {@link #onHandleIntent}, which the system calls when the IntentService
* receives a start request.
* @see android.app.Service#onStartCommand
*/
@Override
public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
onStart(intent, startId);
return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}
在HandleMessage中调用OnHandleIntent抽象函数对任务进行处理,通过重写onHandleIntent依次处理不同intent对应的任务。其中使用StopSelf(msg.arg1)其实就是StopSelf(StartId),就是上面提到的根据startId判断当前startId是不是消息队列里最后一个任务。如果是,就结束服务,否则继续处理后面的任务。而如果调用StopSelf()会立刻停止服务。因此IntentService不需要自己主动停止服务,当处理完所有任务,服务自动停止。
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);
}
}
@WorkerThread
protected abstract void onHandleIntent(@Nullable Intent intent);