美文网首页
Android的消息机制Handler

Android的消息机制Handler

作者: 冯员外_SundarFung | 来源:发表于2017-03-17 10:02 被阅读0次

Android的消息机制就是Handler的运行机制。

先研究明白以下3个问题,有助于对Handler有一个深入的认识。

1.为什么会有Handler呢?

最主要是因为Android不允许非主线程对UI进行操作,子线程去访问网络做一些耗时运算后得到了我们想要的数据,但是尴尬的事情发生了子线程此时手握数据但是它没有权限去改UI。这时Handler伸出了天使的翅膀…Handler从子线程手中接过了数据去把数据给了主线程然后主线程就更新了UI。

题外话

ViewRootImpl的checkThread方法对UI操作做了线程的验证如果不是主线程就是报错“Only the original thread that created a view hierarchy can touch its views”中文翻译就是“越俎代庖”

2.所以Handler是用来干什么的呢?

Handler在Android系统中的作用非常关键,它可以解决子线程不能更新UI的矛盾,同时可以对消息进行发送和处理。

这就好比只有皇上才能发圣旨,但是皇上不会去做具体的事,原因大家都懂,皇上管的是江山社稷,如果他都去做具体的事了,估计国家基本也要完蛋了。所以这些事只能由各个大臣去实施,大臣们在各自州府县衙办事过程中获得了新的事件信息,这些信息想要让天下人都知道怎么办,又不能发圣旨昭告天下,这事只有皇上才能干啊。这时驿站出现了,官老爷们把消息给了驿站,驿站收到信件奏折,快马加鞭把消息传到了宫城内的皇帝手中,皇帝大笔一挥将信息昭告天下。所以脑洞告诉我Handler可以理解为是一个邮驿系统

3.为什么Android不允许非主线程处理UI呢?

因为Android的UI控件不是线程安全的,如果多线程并发访问可能会大致UI控件处于不可预期的状态,你可能想到可以用锁解决,但是用锁首先就是访问UI的逻辑会变得很复杂,其次因为锁会阻塞某些线程的执行而导致降低UI的访问效率。所以最简单且高效的方法就是采用单线程模型来处理UI操作。(该句话出自Android开发艺术探索)这样我们也就根本不用去关心多线程的问题了。另外,开什么玩笑,人人都能发圣旨,那不天下大乱了!

需要注意的是,不允许子线程操作UI并不是不能,如果作死了非要在子线程操作UI也是可以的。因为取决于能不能的关键是ViewRoot在不在该线程中。说一下,更新界面的操作会调用一个invalidate()方法这个方法会去checkThread,而invalidate()就是ViewRootImpl调用的,ViewRootImpl是在Activity的onResume时候通过window getDecorView 后addView时被创建的。所以在onResume前做就不会报错了。这点不多说了,知道子线程也是可以更新UI的就行,别人问子线程能不能更新UI,你就说实际上是能的就OK,别的不用知道了,开发中正常人都不会这么做。

如何使用Handler

在创建Handler的时候,它会跟一个默认的线程进行绑定,这个线程中有一个MessageQueue(消息队列),MessageQueue只是一个消息的存储单元,并不能处理消息,而Looper就是用来做这个的。Looper一直循环的去查是否有新消息,如果有就处理。线程默认是没有Looper的,这需要去创建(用Looper.prepare(),同时在消息相关代码最后调用Looper.loop()才可以进行轮询)。但在UI线程中因为它是ActivityThread,ActivityThread创建的时候就初始化了Looper,所以我们在主线程中可以直接使用Handler.

Handler handler=new Handler(){

public void handlerMessage(Message msg){

mas.what…//发过来的Message在这里被接收到了 可以获取msg的信息

           }

}

主要方法有:

Handler.post(Runnable)  直接运行在UI线程中

Handler.postDelayed(Runnable,long) long代表每隔多少时间去执行这个方法

Handler.sendMessage(Message)

Handler.senMessageDelayed

Handler.removeCallbacks(Runnable); 将该消息在消息列表中移除。

其中所传送的Message以下主要属性
Message.arg (int)可以存放整型数据
Message.obj  (Object)存放Object
Message.what  (int) 指定自定义消息代码,用于接收时候区分
Message.sendToTarget(); 将消息发送给自己,也就是发送给当前的Handler。可以替用Handler.sendMessage();等发送消息的方法。

如果有些情况下没有必去要创建一个Message时,也可以复用系统的Message对象:handler.obtainMessage();会返回一个Message对象。

我们发送消息的话肯定要指定一个消息发送的地址,要知道消息是发送到哪里的,obtainMessage()方法中实际上也就是把消息发送给这个Hnadler自己。方法内部是Message.target来实现的而这个target就是当前的Handler。

Message.sendToTarget()里面实际上也是调用了 target.sendMessage();上面说了target就是Handler自己,所以本质上Message.sendToTarget和Handler.sendMessage();方法是一样的。

Handler原理
MessageQueue

就是消息队列,其中的消息可以添加和删除。Handler发送的消息都汇总到这里来。

Looper

Looper的内部包含了MessageQueue消息队列,所以Looper可以对MessageQueue进行肆无忌惮的轮询,Looper.Loop方法已经启动就开启死循环对MessageQueue进行轮询,如果有消息就处理,没有的话就阻塞。

创建Looper的时候就new了MessageQueue,并把当前的线程给确定了。

Looper(boolean){

       MessageQueue mQueue=new MessageQueue(boolean);

        …….

       mThread = Thread.currentThread();

}
Handler

handler内部与Looper关联,然后Looper内部有MessageQueue,Handler发送的消息都在MessageQueue里

简单说就是handler负责发送消息,Looper负责接收消息并把消息传回给handler自己。

要注意的是如果实在主线程中创建的Handler的话,在handlerMessage方法中不要去写太耗时的方法,否则会出现卡UI的现象。

如果一定要用handler处理耗时操作的话也可以用 HandlerThread,HandlerThread创建了Looper然后关联默认的Handler,解决了多线程并发问题,避免绑定指定looper时候会出现的空指针问题。这样就完全不占用主线程资源

HandlerThread thread=new HandlerThread(“xxxxxthread”);

thread.start();

handler=new Hanlder(thread.getLooper()){

public void handlerMessage(Message msg){

//耗时操作在子线程中执行 操作数据库 下载资源什么的

         }

}

handler.sendMessage……..;//主线程发送消息就行
关于更新UI

除了直接利用handler外runOnUiThread();和view本身的post方法也可以更新UI

private void xxxxUpdateUI(){

        runOnUiThread(new Runnable){

             public void run(){

                //更新UI的方法
                               }
                        }
                  }

handler的post方法以及runOnUiThread()以及view的post方法内部实际上都是封装好的handler发送message的一套方式,其实本质上都是通过handler机制来做的。

相关文章

网友评论

      本文标题:Android的消息机制Handler

      本文链接:https://www.haomeiwen.com/subject/fstenttx.html