HandlerThread和IntentService源码分析

这两个类的原理很简单,前提是在掌握Android 最详细的消息机制解析的基础上。

在上一篇文章的最后提到了如何在子线程中使用Handler,并且讲了HandlerThread的使用,这里回顾一下HandlerThread的使用:

private Handler mHandler;

private void test() {
    HandlerThread ht = new HandlerThread("thread-0");
    ht.start();
    mHandler = new Handler(ht.getLooper()) {
        @Override
        public void handleMessage(@NonNull Message msg) {
            // 处理子线程消息
        }
    };
}
  • 创建HandlerThread对象;
  • 调用HandlerThreadstart()方法;
  • 创建处理子线程事件的Handler

这个顺序是固定的。

HandlerThread源码分析

public class HandlerThread extends Thread {
    int mPriority;
    int mTid = -1;
    Looper mLooper;
    private @Nullable Handler mHandler;

    public HandlerThread(String name) {
        super(name);
        mPriority = Process.THREAD_PRIORITY_DEFAULT;
    }
    
    /**
     * Constructs a HandlerThread.
     * @param name
     * @param priority The priority to run the thread at. The value supplied must be from 
     * {@link android.os.Process} and not from java.lang.Thread.
     */
    public HandlerThread(String name, int priority) {
        super(name);
        mPriority = priority;
    }
}

HandlerThread继承Thread类,它有两个构造方法,通过构造方法中的super(name)可以确定,传入的name就是线程的名称;

当调用了start()方法,虚拟机便会在新线程中执行run()方法,HandlerThreadrun()方法如下:

@Override
public void run() {
    mTid = Process.myTid();
    Looper.prepare();
    synchronized (this) {
        mLooper = Looper.myLooper();
        notifyAll();
    }
    Process.setThreadPriority(mPriority);
    onLooperPrepared();
    Looper.loop();
    mTid = -1;
}

第3行,获取当前线程的线程id;

第4行,Looper.prepare()创建当前线程的Looper对象;

第6行,获取当前线程的Looper对象;

第9行,设置线程的优先级;

第10行,onLooperPrepared()是个protected修饰的空方法;

第11行,Looper.loop()开启子线程消息循环;

第12行,线程执行完毕,即将进入TERMINATED状态,将线程id置为-1;

关于Looper的创建过程在上一篇Android 最详细的消息机制解析已经进行了详细的介绍,不再赘述。这里分析一下,第7行notifyAll()的作用。看下面一段代码:

private Handler mHandler;

private void test() {
    HandlerThread ht = new HandlerThread("thread-0");
    ht.start();
    Looper looper = ht.getLooper();
}

假设test()方法是在主线程中执行,ht.getLooper()是获取子线程的Looper对象,而子线程的Looper对象是在线程run()方法中创建的,这就导致ht.getLooper()的时机可能会早于子线程Looper创建时机。为了确保ht.getLooper()获取到的Looper对象的可靠性,当发现子线程Looper对象未创建时,让主线程处于WAITING状态。

上面的notifyAll()方法在Looper创建后调用,这样主线程就从WAITING状态切换到了RUNNABLE状态,并且可以拿到Looper对象,具体看下HandlerThread#getLooper()方法实现:

public Looper getLooper() {
    if (!isAlive()) {
        return null;
    }
    
    // If the thread has been started, wait until the looper has been created.
    synchronized (this) {
        while (isAlive() && mLooper == null) {
            try {
                wait();
            } catch (InterruptedException e) {
            }
        }
    }
    return mLooper;
}

第8行,判断线程存活并且mLooper == null代表mLooper对象还未初始化,这个时候将调用线程wait(),等待notify()

HandlerThread中也有getHandler()方法,不过这个方法平时用的不多,原因是返回的Handler我们不能处理Handler#sendXXXMessageXX()相关消息,只能用于Handler#postRunnable()

/**
 * @return a shared {@link Handler} associated with this thread
 * @hide
 */
@NonNull
public Handler getThreadHandler() {
    if (mHandler == null) {
        mHandler = new Handler(getLooper());
    }
    return mHandler;
}

IntentService源码分析

Service和IntentService的异同

  • IntentService主要处理耗时操作,Service不建议处理耗时操作,15s会ANR;
  • IntentService执行完后自动停止销毁,Service不会。Service的销毁如下:当通过startService()启动的服务需要通过stopService()或者Service#stopSelf()停止服务器,当通过bindService()启动的服务需要通过unbindService()停止服务,当通过startService()+bindService()启动的服务需要通过stopService()+unbindService()停止服务;
  • IntentService继承自Service
  • 两者使用都必须在AndroidMenifest.xml中注册。

它的实现是基于Service+HandlerThread+Handler

基本使用

定义IntentService

class MyIntentService extends IntentService {
    public MyIntentService(String name) {
        super(name);
    }

    @Override
    protected void onHandleIntent(@Nullable Intent intent) {
        // 处理耗时操作
    }
}

启动Service

Intent intent = new Intent(this,MyIntentService.class);
startService(intent);

源码解析

public abstract class IntentService extends Service {
  
    private volatile ServiceHandler mServiceHandler;
  
    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);
        }
    }
  
    @Override
    public void onCreate() {
        super.onCreate();
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        thread.start();

        mServiceLooper = thread.getLooper();
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }

    @Override
    public void onStart(@Nullable Intent intent, int startId) {
        Message msg = mServiceHandler.obtainMessage();
        msg.arg1 = startId;
        msg.obj = intent;
        mServiceHandler.sendMessage(msg);
    }
   
    @WorkerThread
    protected abstract void onHandleIntent(@Nullable Intent intent); 
  
    @Override
    public void onDestroy() {
        mServiceLooper.quit();
    }
}
  1. Service创建时会创建HandlerThread对象,并且创建用户在子线程中发送消息的Handler;
  2. onStart()中构造一个Message,并通过Handler发送该消息;
  3. ServiceHandlerhandleMessage()收到这个消息并调用onHandleIntent()进行耗时操作处理;
  4. 处理完耗时操作后,通过调用stopSelf(msg.arg1)停止服务;
  5. 服务停止会回调onDestroy()方法,onDestroy()中调用Looper#quit()方法,将消息队列清空;

只要掌握了Handler-Looper-MessageQueue-Message原理,HandlerThreadIntentService的原理掌握起来就很容易了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,560评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,104评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,297评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,869评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,275评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,563评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,833评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,543评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,245评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,512评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,011评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,359评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,006评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,062评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,825评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,590评论 2 273
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,501评论 2 268