Android IntentService全面解析

2018-11-28  本文已影响0人  CyanStone

1. IntentService简介

我们知道,Service用于执行后台任务,而所谓的后台任务,是指跟Activity这种需要跟UI交互组件的生命周期没有关系的任务,所以Service其实跟线程没有半毛钱关系,它的执行也是在主线程中,所以才有了Android ANR触发原理一文中分析的,如果Service的执行时间过长,将触发ANR。
一般,如果需要在Service中执行长时间的耗时操作,标准的写法应该如下:

public class MyService extends Service {         
    //服务执行的操作
    @Override  
    public int onStartCommand(Intent intent, int flags, int startId) {  
        new Thread(new Runnable() {
            public void run() {
                //处理具体的逻辑
                ...
               //服务执行完毕后自动停止
                stopSelf();  
            }
        }).start();        
        return super.onStartCommand(intent, flags, startId);  
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO Auto-generated method stub
        return null;
    }      
 
}

在覆写onStartCommand方法的时候,需要开启子线程,在子线程中执行长时间耗时的操作,执行完毕以后把服务给停止掉。需要跟主线程进行通信的,可以考虑在启动Service的时候把Activity或者ContentProvider组件与Service进行绑定,在onBind方法中需要返回一个IBinder对象,在ServiceConnection对象中的onServiceConnected方法拿到该IBinder对象进行通信,具体的可以参考Android组件系列----Android Service组件深入解析

通过上述分析,我们知道如果想在Service中执行长时间、耗时的操作,就必须开启子线程去执行。Google为了开发者使用方便,对Service组件进行了封装,使得Service具备了工作线程执行的能力,避免了ANR。所以在Service组件的开发中,用户可以自己开启子线程进行控制,也可以直接使用IntentService。


2. IntentService源码分析

废话不多说,我们直接来看IntentService的源码,源码很少,但是我们还是一点点来剖析。先来看下IntentService的继承关系等声明信息:

public abstract class IntentService extends Service {
    ...
    @WorkerThread
    protected abstract void onHandleIntent(@Nullable Intent intent);
}

可以看出IntentService是继承自Service的抽象类,有个抽象方法onHandleIntent需要子类覆写,通过注解我们知道该方法的执行是在子线程中的,具体执行的逻辑下文分析。现在我们来看下IntentService中声明的字段:

//Service中子线程中的Looper对象,volatile修饰,保证其可见性
private volatile Looper mServiceLooper;
//与子线程中Looper关联的Hander对象,volatile修饰,保证其可见性
private volatile ServiceHandler mServiceHandler;
//与子线程HandlerThread相关的一个标识,不重要
private String mName;
//设置Service的标志位,根据它的值来设置onStartCommand的返回值
private boolean mRedelivery;

这里需要特别说明一点,mRedelivery是来处理onStartCommand返回值的一个标志位参数,具体的返回参数我们下文分析,先看下onStartCommand的返回值在Service已经定义了几种:

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);
    }
}

从这里可以看出,抽象方法onHandlerIntent方法是在与ServiceHandler相关的线程执行的,具体是在哪个线程,我们在下文进行分析。需要注意的是,在消息被处理完后,Service会被停止。下面先来看构造方法:

public IntentService(String name) {
    super();
    mName = name;
}

从构造方法可以看出,IntentService的构造方法中只提供了带参数的构造方法,所以子类的构造方法中,必须调用这个构造方法,并传入一个Name字段,该字段用于标识子线程,调试的时候用,初始化HandlerThread的时候会用到。下面我们来分析on

@Override
public void onCreate() {
    super.onCreate();
    //初始化了一个HandlerThread,并开启了子线程
    HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
    thread.start();
    //从HandlerThread中拿到了looper对象,并用它来初始化Handler对象,所以
    //ServiceHandler 中调用的onHandleIntent方法,是在子线程中执行的
    mServiceLooper = thread.getLooper();
    mServiceHandler = new ServiceHandler(mServiceLooper);
}

@Override
public void onStart(@Nullable Intent intent, int startId) {
    //获取一个Message对象,并把startId和Intent保存到Message中,并把这个Message发送到子线程中执行
    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) {
    //注释写的很清楚,不建议子类覆写这个方法,而是应该把想要实现的逻辑放到onHandleIntent方法中
    onStart(intent, startId);
    return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}

@Override
public void onDestroy() {
    //在Service销毁的时候,停止消息队列
    mServiceLooper.quit();
}

源码其实很简单,这里说明以下几点:

/**
 * @deprecated Implement {@link #onStartCommand(Intent, int, int)} instead.
 */
@Deprecated
public void onStart(Intent intent, int startId) {
}

3. 用法与注意事项

  1. 子类继承IntentService,并在子类的构造方法中调用父类带参构造方法;
  2. 实现onHandleIntent方法逻辑,完成需要在子线程中完成的逻辑,可利用Intent进行传值;
  3. 其他如与组件绑定、开启、停止等,与Service一致;

具体的用法demo这里就不写了,跟Service差不多,只要把想要执行的逻辑放在onHandleIntent中就可以了,省去了开启子线程和stopSelf的操作。

参考链接
Android组件系列----Android Service组件深入解析
Android HandlerThread全面解析
Android异步消息处理机制源码剖析

上一篇下一篇

猜你喜欢

热点阅读