IntentService源码分析

简介: 我们知道Service是运行在主线程的,主线程中不能进行耗时操作,否则会发生ANR。Service中的发生ANR的超时时间是20s。有时候我们需要应用在后台默默做一些任务,例如上传文件等。如果我们采用Service,则需要我们自己手动开启新的线程。如果我们不想 自己开启线程怎么办,IntentService就出现了。

我们知道Service是运行在主线程的,主线程中不能进行耗时操作,否则会发生ANR。Service中的发生ANR的超时时间是20s。
有时候我们需要应用在后台默默做一些任务,例如上传文件等。如果我们采用Service,则需要我们自己手动开启新的线程。如果我们不想 自己开启线程怎么办,IntentService就出现了。

基本使用

public class TestService extends IntentService {
    public YourUploadService() {
        super("TestService");
    }
    @Override
    protected void onHandleIntent(Intent intent) {
        //耗时操作
    }
}

使用IntentService需要我们继承IntentService,并重写onHandleIntent方法,在onHandleIntent中进行耗时操作。

源码分析

IntentService的源码比较简单,这里就直接全部贴出

public abstract class IntentService extends Service {
    private volatile Looper mServiceLooper;
    private volatile ServiceHandler mServiceHandler;
    private String mName;
    private boolean mRedelivery;
    //子线程的Handler
    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;
    }
    //  设置onStartComand的返回策略
    public void setIntentRedelivery(boolean enabled) {
        mRedelivery = enabled;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");//创建一个HandlerThread
        thread.start();
        mServiceLooper = thread.getLooper();//获取子线程的Looepr
        mServiceHandler = new ServiceHandler(mServiceLooper);//创建ServiceHandler,子线程的Handler
    }

    @Override
    public void onStart(@Nullable Intent intent, int startId) {
        Message msg = mServiceHandler.obtainMessage();
        msg.arg1 = startId;
        msg.obj = intent;
        mServiceHandler.sendMessage(msg);//向子线程的Handler的发送消息,进行任务处理
    }

    @Override
    public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        onStart(intent, startId);
        return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;//START_REDELIVER_INTENT:重传Intent。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务,并将Intent的值传入。START_NOT_STICKY:如果在执行完onStartCommand后,服务被异常kill掉,系统不会自动重启该服务
    }

    @Override
    public void onDestroy() {
        mServiceLooper.quit();
    }

    @Override
    @Nullable
    public IBinder onBind(Intent intent) {
        return null;
    }

    @WorkerThread
    protected abstract void onHandleIntent(@Nullable Intent intent);
}

总结

我们可以看到IntentService本质是一个Service,继承了Service。与Service的区别是内部通过维护了一个HandlerThread创建的线程,并与Handler搭配使用,达到在子线程中处理任务的目的。

相关文章
|
5月前
|
Java Android开发 容器
事件分发源码分析
事件分发源码分析
52 2
|
Android开发
Android多线程之IntentService
IntentService继承自Service,所以IntentService也是四大组件之一,IntentService内部封装了HandlerThread线程 (只有一个线程) 来按顺序处理异步任务
|
Android开发 容器 Python
DialogFragment源码分析
目录介绍 1.最简单的使用方法 1.1 官方建议 1.2 最简单的使用方法 1.3 DialogFragment做屏幕适配 2.源码分析 2.1 DialogFragment继承Fragment 2.
1183 0
IntentService详解
IntentService是什么? 这篇文章是之前就写好的,一直没有整理出来,这几天有空正好整理发布一下。 我们知道Service可以让我们在后台处理一些事情,但是Service实际上也是主线程,所以执行长耗时任务时依然会ANR,只不过ANR触发时间要比前台长。一般我们会在Service中开启一个子线程去完成耗时任务。 而IntentService就是解决这个问题的,它是Service的一个抽象子类,需要实现onHandleIntent,代码在这个函数中执行。它与Service最大的不同就是默认开启一个子线程,而onHandleIntent就是在子线程中执行的。
148 0
线程池源码分析_01 FutureTask源码分析
文章参考:硬核手撕Java线程池FutureTask源码
|
消息中间件 数据库
HandlerThread原理与应用
HandlerThread  在理解了Handler的原理之后,我们知道在一个子线程中创建一个Handler不能缺少了Looper.prepare()和Looper.loop()两个方法,具体的原因这里不再赘述,不熟悉原理的可以先看下另一篇文章Handler的原理解析. 本篇文章主要是讲解HandlerThread的使用的。
1206 0
|
Android开发
Activity 的启动流程源码剖析(三)
简单分析 Activity 的启动流程(三) 这一篇主要看 ApplicationThread 相关的流程 源码的版本 Android 27 V4 27.
1126 0