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搭配使用,达到在子线程中处理任务的目的。

相关文章
|
7月前
|
存储 算法 安全
ConcurrentLinkedQueue 的实现原理分析
ConcurrentLinkedQueue 的实现原理分析
32 0
|
8月前
|
Java Android开发 容器
事件分发源码分析
事件分发源码分析
61 2
|
Android开发
Android多线程之IntentService
IntentService继承自Service,所以IntentService也是四大组件之一,IntentService内部封装了HandlerThread线程 (只有一个线程) 来按顺序处理异步任务
|
Android开发
Android之HandlerThread源码分析和简单使用(主线程和子线程通信、子线程和子线程通信)
Android之HandlerThread源码分析和简单使用(主线程和子线程通信、子线程和子线程通信)
233 0
Android之HandlerThread源码分析和简单使用(主线程和子线程通信、子线程和子线程通信)
IntentService详解
IntentService是什么? 这篇文章是之前就写好的,一直没有整理出来,这几天有空正好整理发布一下。 我们知道Service可以让我们在后台处理一些事情,但是Service实际上也是主线程,所以执行长耗时任务时依然会ANR,只不过ANR触发时间要比前台长。一般我们会在Service中开启一个子线程去完成耗时任务。 而IntentService就是解决这个问题的,它是Service的一个抽象子类,需要实现onHandleIntent,代码在这个函数中执行。它与Service最大的不同就是默认开启一个子线程,而onHandleIntent就是在子线程中执行的。
159 0
|
Android开发
Android之HandlerThread源码分析和简单使用(主线程和子线程通信、子线程和子线程通信1)
Android之HandlerThread源码分析和简单使用(主线程和子线程通信、子线程和子线程通信)
200 0
线程池源码分析_01 FutureTask源码分析
文章参考:硬核手撕Java线程池FutureTask源码
|
Java 程序员 算法
基于接口回调详解JUC中Callable和FutureTask实现原理
Callable接口和FutureTask实现类,是JUC(Java Util Concurrent)包中很重要的两个技术实现,它们使获取多线程运行结果成为可能。它们底层的实现,就是基于接口回调技术。接口回调,许多程序员都耳熟能详,这种技术被广泛应用于异步模块的开发中。
1422 0
|
消息中间件 数据库