IntentService是继承于Service并处理异步请求的一个类,在IntentService内有一个工作线程来处理耗时操作,启动IntentService的方式和启动传统Service一样,同时,当任务执行完后,IntentService会自动停止,而不需要我们去手动控制。另外,可以启动IntentService多次,而每一个耗时操作会以工作队列的方式在IntentService的onHandleIntent回调方法中执行,并且,每次只会执行一个工作线程,执行完第一个再执行第二个,以此类推。

而且,所有请求都在一个单线程中,不会阻塞应用程序的主线程(UI Thread),同一时间只处理一个请求。

IntentService有什么好处呢?

1)我们省去了在Service中手动开线程的麻烦,

2)当操作完成时,我们不用手动停止Service。

接下来让我们来看看如何使用,写一个Demo来模拟两个耗时操作,Operation1与Operation2,先执行1,2必须等1执行完才能执行2:

新建工程,新建一个继承IntentService的类,我这里是IntentServiceDemo.java

  1. public class IntentServiceDemo extends IntentService {
  2. public IntentServiceDemo() {
  3. //必须实现父类的构造方法
  4. super("IntentServiceDemo");
  5. }
  6. @Override
  7. public IBinder onBind(Intent intent) {
  8. System.out.println("onBind");
  9. return super.onBind(intent);
  10. }
  11. @Override
  12. public void onCreate() {
  13. System.out.println("onCreate");
  14. super.onCreate();
  15. }
  16. @Override
  17. public void onStart(Intent intent, int startId) {
  18. System.out.println("onStart");
  19. super.onStart(intent, startId);
  20. }
  21. @Override
  22. public int onStartCommand(Intent intent, int flags, int startId) {
  23. System.out.println("onStartCommand");
  24. return super.onStartCommand(intent, flags, startId);
  25. }
  26. @Override
  27. public void setIntentRedelivery(boolean enabled) {
  28. super.setIntentRedelivery(enabled);
  29. System.out.println("setIntentRedelivery");
  30. }
  31. @Override
  32. protected void onHandleIntent(Intent intent) {
  33. //Intent是从Activity发过来的,携带识别参数,根据参数不同执行不同的任务
  34. System.out.println("currentThread()=" + Thread.currentThread().getName());
  35. String action = intent.getExtras().getString("param");
  36. if (action.equals("oper1")) {
  37. System.out.println("Operation1");
  38. }else if (action.equals("oper2")) {
  39. System.out.println("Operation2");
  40. }
  41. try {
  42. Thread.sleep(2000);
  43. } catch (InterruptedException e) {
  44. e.printStackTrace();
  45. }
  46. }
  47. @Override
  48. public void onDestroy() {
  49. System.out.println("onDestroy");
  50. super.onDestroy();
  51. }
  52. }

我把生命周期方法全打印出来了,待会我们来看看它执行的过程是怎样的。接下来是Activity,在Activity中来启动IntentService:

  1. public class TestActivity extends Activity {
  2. /** Called when the activity is first created. */
  3. @Override
  4. public void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.main);
  7. //可以启动多次,每启动一次,就会新建一个work thread,但IntentService的实例始终只有一个
  8. //Operation 1
  9. Intent startServiceIntent = new Intent("com.test.intentservice");
  10. Bundle bundle = new Bundle();
  11. bundle.putString("param", "oper1");
  12. startServiceIntent.putExtras(bundle);
  13. startService(startServiceIntent);
  14. //Operation 2
  15. Intent startServiceIntent2 = new Intent("com.test.intentservice");
  16. Bundle bundle2 = new Bundle();
  17. bundle2.putString("param", "oper2");
  18. startServiceIntent2.putExtras(bundle2);
  19. startService(startServiceIntent2);
  20. }
  21. }

最后,别忘了配置Service,因为它继承于Service,所以,它还是一个Service,一定要配置,否则是不起作用的

  1. <service android:name=".IntentServiceDemo">
  2. <intent-filter >
  3. <action android:name="com.test.intentservice"/>
  4. </intent-filter>
  5. </service>

最后来看看执行结果:

img

从结果可以看到,onCreate方法只执行了一次,而onStartCommand和onStart方法执行了两次,开启了两个Work Thread,这就证实了之前所说的,启动多次,但IntentService的实例只有一个,这跟传统的Service是一样的。Operation1也是先于Operation2打印,并且我让两个操作间停顿了2s,最后是onDestroy销毁了IntentService。

IntentService 源码分析

  1. @Override
  2. public void onCreate() {
  3. super.onCreate();
  4. HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
  5. thread.start();
  6. mServiceLooper = thread.getLooper();
  7. mServiceHandler = new ServiceHandler(mServiceLooper);
  8. }

源码可知:

1)实际上是使用了一个 HandlerThread 来维护线程的,

2) HandleThread 中,内部已经维护一个 Looper,这里直接使用 HandlerThread 的 Looper 对象,便于在 IntentService 中去维护消息队列,

3)创建的 mServiceHandler 是属于 HandleThread 这个 WorkerThread 的。

  1. private final class ServiceHandler extends Handler {
  2. public ServiceHandler(Looper looper) {
  3. super(looper);
  4. }
  5. @Override
  6. public void handleMessage(Message msg) {
  7. onHandleIntent((Intent)msg.obj);
  8. stopSelf(msg.arg1);
  9. }
  10. }

源码可知:

1)直接把消息交给 onHandleIntent() 方法去执行具体的业务逻辑

2)执行完成之后,立即调用 stopSelf() 方法停止自己

接下来分析start源码

  1. @Override
  2. public void onStart(Intent intent, int startId) {
  3. Message msg = mServiceHandler.obtainMessage();
  4. msg.arg1 = startId;
  5. msg.obj = intent;
  6. mServiceHandler.sendMessage(msg);
  7. }
  8. @Override
  9. public int onStartCommand(Intent intent, int flags, int startId) {
  10. onStart(intent, startId);
  11. return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
  12. }

源码可知

1)在 onStartCommand() 中直接调用了 onStart() 方法

2)而上面 stopSelf() 方法使用的 startId 来停止当前的此次任务服务。

3)而 Service 如果被启动多次,就会存在多个 startId ,当所有的 startId 都被停止之后,才会调用 onDestory() 自我销毁。

我们在看看HandlerThread启动之后的源码

  1. @Override
  2. public void run() {
  3. mTid = Process.myTid();
  4. Looper.prepare();
  5. synchronized (this) {
  6. mLooper = Looper.myLooper();
  7. notifyAll();
  8. }
  9. Process.setThreadPriority(mPriority);
  10. onLooperPrepared();
  11. Looper.loop();
  12. mTid = -1;
  13. }

源码可知

run方法里面添加了锁,这也解释了为什么多次 start 同一个 IntentService 它会顺序执行,全部执行完成之后,再自我销毁。