Android-Handler机制

参考:https://mp.weixin.qq.com/s/AvpOq6eh-IBDJj2SXa_wXw
https://www.jianshu.com/p/25a05bf42e05

Handler是我们项目里常用的更新UI的方式了,如:子线程获取数据通过Handler.sendMessage()发送消息,在UI线程Handler.handleMessage()就会调用,执行相应处理,同时Handler机制也是我们面试常问到的内容,我们有必要做一个总结。

什么是HanlerThread类?

如果一个线程创建了Handler对象,同时也创建了Loop对象拥有MessageQueue,那么我们就可以说这个线程是HandlerThread.

Handler机制主要的几个类

  • Handler:用来发送消息:sendMessage等多个方法,并实现handleMessage()方法处理回调
  • Message:消息实体,发送的消息即为Message类型。
  • MessageQueue:消息队列,是一个单链表,用于存储Message。发送消息时,消息入队列,然后Looper会从这个MessageQueue取出消息进行处理。
  • Loop:与线程绑定,绑定的线程用来处理消息。loop()方法是一个死循环,一直从MessageQueen里取出消息进行处理。注意:Loop绑定的可以是子线程也可以是主线程

那么这里就有一个问题了:loop()方法是一个死循环,为什么不会被卡死呢?
答:简单的说,就是当MessageQueue中没有Message时,线程会挂起,一有消息,就会唤醒线程处理消息。
这几个类非常重要,也是我们这次要重点讲解的。

Handler

Handler可以发送什么
handler有两个工作,一是发送任务或者消息;二是处理消息或者执行任务。Handler和MessageQueue关系密切,Handler不单单能发送Message到MessageQueue,也可以发送Runnable.或者这么说MessageQueue不仅仅放Message也存放Runnable.

Handler的触发线程
通常Handler都是在创建线程创建,在非创建线程发送的(当然也可以在同一个线程中创建发送),而handler处理消息或执行任务,则是在创建自己的线程(创建线程)中执行的。

创建Handler
handler和looper的创建并不是ui线程独有的。任何一个普通的线程,都可以创建自己的looper,创建自己的handler。但是有一点需要注意,创建handler前,必须先创建looper。
创建代码如下:

  /**
     * 在子线程创建Handler
     */
    public Handler mHandler;
    
    class LooperThread extends Thread {
        public void run() {
            //调用Looper.prepare();语句来为当前线程new一个looper对象,将当前线程(子线程)与new出来的Looper关联
            Looper.prepare();
            mHandler = new Handler() {
                public void handleMessage(Message msg) {

                }
            };
            //Looper.loop()开启循环处理消息
            Looper.loop();
        }
    }

这里又有一个问题了?我们平时都是直接在Activit里new Handler(),为什么没有创建Looper对象呢?
答:主线程在启动的时候,默认就会调用Looper.prepareMainLooper() 创建looper,所以,我们可以在主线程直接创建handler,不需要手动先创建looper。

Handler发送Runnable
我们上面说了,Handler不单单可以发送Message还可以发送Runnabe,代码如下:

 /**
     * Handler发送Runnable
     */
    public void sendRunnable(){
        mHandler.postDelayed(new Runnable() {
            @Override
            public void run() {
    
            }
        },3000);
    }

我们顺着看一眼源码:

 public final boolean post(Runnable r)
    {
       return  sendMessageDelayed(getPostMessage(r), 0);
    }

 private static Message getPostMessage(Runnable r) {
        Message m = Message.obtain();
//这里m.callback就是一个Runnable
        m.callback = r;
        return m;
    }

用getPostMessage 把runnable包装成一个message,message的callback就是runnable。message.callback其实就是对runnable的封装,因此,分辨一个message是不是runnable,其实只要看message的callback是否为空,如果为空,就是普通的message,否则就是一个runnbale。

Handler处理消息
我们刚才说来了Handler可以发送Message,可以发送Runnabe,那么他是如何区分来处理消息的呢?我们来看一下源码Handler类中的dispatchMessage这个方法:

/**
     *在这里处理系统消息
     */
    public void dispatchMessage(Message msg) {
        if (msg.callback != null) {
            handleCallback(msg);// handler.post(runnable)时走这里
        } else {
            if (mCallback != null) {
 // handler = new Handler(callback)时走这里
                if (mCallback.handleMessage(msg)) {
                    return;
                }
            }
// handler = new Handler()时走这里
            handleMessage(msg);
        }
    }

根据handler发送消息的类型,分成2种情况:

handler发送了一个message到MessageQueue

handler发送了一个runnbale到MessageQueue

根据前面提到的,message.callback其实就是对runnable的封装,所以,如果handler是发送了一个runnable到MessageQueue,那么就会执行这个runnable。

如果handler是发送了一个message到MessageQueue,那么又细分为2种情况

  • handler创建时设置了callback, 即handler = new Handler(callback);

  • handler创建时未设置callback,即handler = new Handler();

如果设置了callback,那么message会先被callback处理。

如果callback返回true,说明处理完成,不会再传给handler.handleMessage了。

如果callback返回false,说明处理未完成,会再把message传给handler.handleMessage继续处理。

如果未设置callback,message会直接传给handler.handleMessage处理

Message

Message如何产生
Message一般是通过构造函数也就是new的方式产生,也可以通过Message.obtain()或Handler.obtainMessage()方式产生,建议使用Message.obtain(),因为此方法获取的Message对象是从可回收的消息对象池里面直接获取的,有利于性能提高。
当然handler.sendEmptyMessage(),可以发送一个空消息到MessageQueue,不需要构造一个message对象。

Message发送时机

  • handler.sendMessage():
    Message发送时机的发送时机一般是由Handler决定的,比如我们常用的handler.sendMessage(Message message)方法,是把message放到MessageQueue的尾部排队,遵循先进先出的原则一个一个取消息发送。
  • handler.sendMessageAtFrontOfQueue():
    把message放到message queue的头部,消息可以马上被处理。
  • handler.sendMessageAtTime():
    不马上发送消息到message queue,而是在指定的时间点发送。
  • handler.sendMessageDelayed():
    不马上发送消息到message queue,而是在指定的时延后再发送。
MessageQueue

MessageQueue 就是用来存储Message的,里面的Message默认先进先出。

Looper

Looper会从MessageQueue中取出Message,然后派发给handler处理。
Looper的创建
Loop.prepare();//此方法创建Looper对象,一个线程对应一个Looper对象
Looper的获取

  //获取主线程中的Looper
        Looper mLoop=Looper.getMainLooper();
   //获取子线程或者Handler构造方法中的looper
                Looper mLooper=Looper.myLooper();

Looper派发消息
我们知道Looper通过 Looper.loop();来循环从MessageQueue发送新增Message.那么他是如何将Message和Handler对应呢,确保这个Message就是发送给这个Hander呢?
我们看一看loop方法:

image.png

每个msg都有一个target属性,这个target就是发送消息的handler,派发message,就是派发给msg.target对象。
loop方法并不是无限循环的,一旦message queue为空,就会结束,以免长期占用cpu资源。

总结:
handler不是独立存在的,一个handler,一定有一个专属的线程,一个消息队列,和一个looper并与之关联。
Handler机制实际是生产者和消费者模式,handler是生产者,生产消息然后添加到messagequeue;looper是消费者,从messagequeue里取message。
Handler,Looper,MessagQueue的工作流程是:

  • handler发送消息到message queue,这个消息可能是message,也可能是runnable,还有可能使空消息
  • looper负责从messagequeue取message
  • looper把message通过dispatchMessage方法发送给给handler
  • handler处理消息(handleMessage或者执行runnable)
    他们的关系可以用一张图说明:


    image.png

    告辞!

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

推荐阅读更多精彩内容