「这是我参与11月更文挑战的第1天,活动详情查看:2021最后一次更文挑战」
IntentService是什么?
这篇文章是之前就写好的,一直没有整理出来,这几天有空正好整理发布一下。
我们知道Service可以让我们在后台处理一些事情,但是Service实际上也是主线程,所以执行长耗时任务时依然会ANR,只不过ANR触发时间要比前台长。一般我们会在Service中开启一个子线程去完成耗时任务。
而IntentService就是解决这个问题的,它是Service的一个抽象子类,需要实现onHandleIntent,代码在这个函数中执行。它与Service最大的不同就是默认开启一个子线程,而onHandleIntent就是在子线程中执行的。
所以IntentService就是一个自带子线程的Service。
那么它是如何实现的,我们通过它的源码来简单分析一下。
线程的创建
IntentService的源码其实不多,先来看看它的onCreate函数
@Override
public void onCreate() {
super.onCreate();
HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
thread.start();
mServiceLooper = thread.getLooper();
mServiceHandler = new ServiceHandler(mServiceLooper);
}
在onCreate中会创建并启动一个HandlerThread。这个HandlerThread是Thread的一个子类,通过它的源码可以看到实际上就是默认绑定开启了looper,它的run函数如下:
@Override
public void run() {
mTid = Process.myTid();
Looper.prepare();
synchronized (this) {
mLooper = Looper.myLooper();
notifyAll();
}
Process.setThreadPriority(mPriority);
onLooperPrepared();
Looper.loop();
mTid = -1;
}
可以看到绑定并启动了一个looper,而它的其他函数则都与looper和handler的操作有关,这里不一一看了。
回到onCreate中,接下来的代码则是通过HandlerThread的looper创建了一个ServiceHandler,这是一个内部类,源码如下:
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);
}
}
可以看到在handleMessage中调用了onHandleIntent,这样函数中的代码就在这个HandlerThread的子线程中运行。
然后执行了stopSelf函数,这个函数是Service的,用于停止服务的,与context的stopService效果是一样的。
也就是说IntentService执行onHandleIntent后就会试图停止服务,但是这里还有一些逻辑,注意传參是msg.arg1,这个很重要,后面会再详谈。
线程运行
通过上面我们知道,IntentService创建时开启一个线程并启动一个looper,并且通过ServiceHandler来执行代码。但是想要执行onHandleIntent,一定需要sendMessage,那么在哪send的呢?
答案是在Service的start周期中,如下:
@Override
public void onStart(@Nullable Intent intent, int startId) {
Message msg = mServiceHandler.obtainMessage();
msg.arg1 = startId;
msg.obj = intent;
mServiceHandler.sendMessage(msg);
}
可以看到在onStart中组合了一个msg并send,这样就实现了服务一启动就自动执行onHandleIntent。
注意msg的arg1参数赋值是startId,结合上面所说的就是执行onHandleIntent后会执行stopSelf(startId)。
上面我们说过stopSelf会试图停止服务,为什么是试图而不是一定,关键就是startId这里。
只有当stopSelf函数的startId与Service当前的startId相同,才会停止服务,主要是针对多次startService的情况。
当我们多次startService的时候,startId会自动递增,并且保存最后一个startId。
所以多次startService,onHandleIntent会执行多次,因为前几次执行到stopSelf时startId不同,只有当最后一个执行完后才真正的停止服务。
但是注意,在onBind函数中并没有sendMessage,所以IntentService需要使用start的方式,bind的方式由于不会走onStart这个周期,所以onHandleIntent不会执行。
退出线程
在创建线程时默认开启了looper,looper其实就是一个死循环,所以这个线程会一直阻塞。那么IntentService如何退出这个线程?
上面我们知道最后通过stopSelf停止服务,但是还没有看到对线程的操作。但是当我们停止服务时,会执行onDestroy函数,来看看源码:
@Override
public void onDestroy() {
mServiceLooper.quit();
}
这里将looper退出了,这样线程循环结束了线程就退出了。所以IntentService停止线程就会退出,如果多次启动service,那么最后一个执行完才会退出线程,这样也保证了任务全部被执行。
总结
IntentService其实很简单,就是内部实现了一个使用Handler机制的子线程而已,但是它使用起来方便了很多。