Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 326|回复: 0

[默认分类] Android中的Handler的机制与用法详解

[复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-7-4 11:28:34 | 显示全部楼层 |阅读模式


      
       
        目录结构
       
       
        Android中的Handler的机制与用法详解,什么是Handler,如何传递 Message,传递 Runnable 对象,传递 Callback 对象,Handler 原理是什么?Handler 与 Looper、MessageQueue 的关系,HandlerThread是什么?Android 中更新 UI 的几种方式。
       
       1. 了解几个概念
       
       
        很多Android初学者对Android 中的Handler不是很明白,其实Google参考了Windows的消息处理机制,在Android系统中实现了一套类似的消息处理机制。
        在下面介绍Handler机制前,首先得了解以下几个概念:
       
       
    1. Message
    复制代码
    消息,理解为线程间通讯的数据单元。例如后台线程在处理数据完毕后需要更新UI,则可发送一条包含更新信息的Message给UI线程。
       
    1. Message Queue
    复制代码
    消息队列,用来存放通过Handler发布的消息,按照先进先出执行。
       
    1. Handler
    复制代码
    Handler是Message的主要处理者,负责将Message添加到消息队列以及对消息队列中的Message进行处理。
       
    1. Looper
    复制代码
    循环器,扮演Message Queue和Handler之间桥梁的角色,循环取出Message Queue里面的Message,并交付给相应的Handler进行处理。
       
    1. 线程
    复制代码
    UI thread 通常就是main thread,而Android启动程序时会替它建立一个Message Queue。每一个线程里可含有一个Looper对象以及一个MessageQueue数据结构。在你的应用程序里,可以定义Handler的子类别来接收Looper所送出的消息。
       
        2. 什么是Handler
       
       
       
       
         Handler 是 Android 给我们提供来更新 UI 的一套机制,也是一套消息处理的机制,我们可以发送消息,也可以通过它来处理消息,Handler 在我们的 framework 中是非常常见的。
         Android 在设计的时候,就封装了一套消息创建、传递、处理机制,如果不遵循这样的机制就没有办法更新 UI 信息,就会抛出异常信息。
       
        3. Handler用法
        3.1 传递message
       
         用于接受子线程发送的数据,并用此数据配合主线程更新 UI。有以下方法:
       
    1. [code]post(Ruannable);
    2. postAtTime(Runnable, long);
    3. postDelayed(Runnable long);
    复制代码
    [/code]
         post类方法允许你排列一个 Runnable 对象到主线程队列中。
         3.2 传递Runnable对象
       
       
         用于通过 Handler 绑定的消息队列,安排不同操作的执行顺序,主要有以下方法:
       
    1. [code]sendEmptyMessage(int);
    2. sendMessage(Message);
    3. sendMessageAtTime(Message, long);
    4. sendMessageDelayed(Message, long);
    复制代码
    [/code]
         sendMessage 类方法,允许你安排一个带数据的 Message 对象到队列中,等待更新。
         
          使用 Handler 在子线程中向 UI 线程发送一个消息进行 UI 的更新
          创建一个 Message,
    1. Message
    2. msg = new Message(); msg.arg1 = 88;
    复制代码
         
    1. handler.sendMessage(msg); msg.obj = xxx;
    复制代码
    可以传递一个对象
          当然不一定要用 new 一个 Message,也可以复用系统的 message 对象
    1. Message msg = handler.obtainMessage();
    复制代码
         
         3.3 传递Callback对象
       
       
         Callback 用于截获 handler 发送的消息,如果返回 true 就截获成功不会向下传递了。
       
    1. [code]public Handler mHandler = new Handler(new Handler.Callback() {
    2.     @Override
    3.     public boolean handleMessage(Message msg) {
    4.         // TODO Auto-generated method stub
    5.         Toast.makeText(getApplicationContext(), "HandleMessage 1", Toast.LENGTH_SHORT).show();
    6.         return true;
    7.     }
    8. }) {
    9.     public void handleMessage(Message msg) {
    10.         // TODO Auto-generated method stub
    11.         Toast.makeText(getApplicationContext(), "handleMessage 1", Toast.LENGTH_SHORT).show();
    12.     };
    13. }
    复制代码
    [/code]
         上面的示例中,第一个有返回值的 handlerMessage 方法是 Callback 的回调,如果返回true,则不执行下面的 handlerMessage 方法,从而达到拦截 handler 发送的消息的目的,如果返回 false,则会继续执行 handlerMessage 方法。
         4. Handler原理
       
        4.1 Android为什么设计成只能通过Handler机制更新UI呢?
       
         最根本的目的就是解决多线程并发的问题,假设在一个 Activity 当中,有多个线程去更新 UI,并且对更新的 UI 的操作进行枷锁处理的话又会产生什么样的问题呢? 那就是性能下降,Handler 通过消息队列,保证了消息处理的先后有序。
         鉴于以上问题的考虑,Android 给我们提供了一套更新 UI 的机制,我们只要使用一套机制就好,所有的更新 UI 的操作都是在主线程中轮询处理。
         4.2 Handler的原理是什么?
       
       
         
          Handler 封装了消息的发送(主要包括消息发送给谁) Looper:
          
            内部包含一个消息队列也就是 MessageQueue,所有 Handler 发送的消息都走向这个队列。
            Looper.loop()方法,就是一个 for 死循环,不断的从 MessageQueue 取消息,如果有消息就处理消息,没有消息就阻塞。
          
          MessageQueue,就是一个消息队列,可以添加消息,处理消息。
          Handler 也不难,比较简单,在构造 Handler 时候内部会跟 Looper 进行关联,通过 Looper.myLooper() 获取到 Looper,找到 Looper 也就找到了 MessageQueue。在 Handler 中发送消息,其实是向 MessageQueue 队列中发送消息。
         
         4.3 handler与Looper、MessageQueue的关系?
       
       
         这一小结:handler 负责发送消息,Looper 负责接收 Handler 发送消息,并直接把消息回传给 handler 自己,MessageQueue 就是一个存储消息的容器。
         
         如上图所示,一个线程中只有一个 Looper 实例,一个 MessageQueue 实例,可以有多个 Handler 实例。
         下图展示了 Handler、MessageQueue、Looper 之间是如何协作的。
         
         下图是一个“我要上厕所”的形象图解:
         
         5. Handler与子线程
       
        5.1 自定义与线程相关的Handler
       
       
    1. [code]class MyThread extends Thread {
    2.     public Handler handler;
    3.     @Override
    4.     public void run() {
    5.         Looper.prepare(); //new 一个Looper对象
    6.         handler = new Handler() { //拿到当前线程的 Looper 对象
    7.             @Override
    8.             public void handlerMessage(Message msg) {
    9.                 // TODO Auto-generated method stub
    10.                 System.out.println("current thread:" + Thread.currentThread());
    11.             }
    12.         };
    13.         Looper.loop();//开始死循环处理消息
    14.     };
    15. }
    复制代码
    [/code]
         一般 UI 主线程中不要执行一些耗时的操作,这样就可以通过子线程消息来处理耗时操作。
         5.2 HandlerThread是什么?
       
       
         HandlerThread 继承于 Thread,所以它本质就是个 Thread。与普通的 Thread 的差别就在于,它有个 Looper 成员变量。这个 Looper 其实就是对消息队列以及队列处理逻辑的封装,简单来说就是消息队列+消息循环。
         当我们需要一个工作线程,而不是把它当作一次性消耗品,用过即废的话,就可以使用它。
       
    1. [code]private Handler mHandler = null;
    2. private HandlerThread mHandlerThread = null;
    3. private void sendRunnableToWorker(Ruannable run) {
    4.     if (null == mHandlerThread) {
    5.         mHandlerThread = new HandlerThread("WorkerThread");
    6.         // 给工作者线程低优先级
    7.         mHandlerThread.setPriority(Thread.MIN_PRIORITY);
    8.         mHandlerThread.start();
    9.     }
    10.     if (null == mHandler) {
    11.         mHandler = new Handler(mHandlerThread.getLooper());
    12.     }
    13.     mHandler.post(run);
    14. }
    复制代码
    [/code]
         6. 主线程与子线程之间的信息交互
       
       
       
    1. [code]//创建主线程的Handler
    2. private Handler mHandler = new Handler() {
    3.     public void handleMessage(Message msg) {
    4.         Message mssage = new Message();
    5.         System.out.println("main Handler");
    6.         //向子线程发送消息
    7.         threadHandler.sendMessageDelayed(message, 1000);
    8.     };
    9. };
    10. //创建子线程的 handler
    11. private Handler threadHandler;
    12. @Override
    13. protected void onCreate(Bundle saveInstanceState) {
    14.     super.onCreate(savedInstanceState);
    15.     setContentView(R.layout.main);
    16.     HandlerThread thread = new HandlerThread("handlerThread");
    17.     //创建子线程的 handler
    18.     threadHandler = new Handler(thread.getLooper()) {
    19.         public void handlerMessage(Message msg) {
    20.             Message message = new Message();
    21.             //向主线程发送消息
    22.             mHandler.sendMessageDelayed(message, 1000);
    23.         };
    24.     };
    25. }
    复制代码
    [/code]
         7. Android中更新UI的几种方式
       
       
         Android 中更新 UI 的 4 种方式:
         
          runOnUiThread
          handler 的 post
          handler 的 sendMessage
          View 自身的 post
         
         

       
       
         

       
       
         

       
       
         

       
       
       
       

       
      
      
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-24 07:24 , Processed in 0.486013 second(s), 46 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表