Android消息机制

作者: 寒_蝉 | 来源:发表于2019-05-04 17:55 被阅读0次

    基本概念

    1. 消息:Message</br>
      消息(Message)代表一个行为(what)或者一串动作(Runnable),每一个消息在加入消息队列时,都有明确的目标(Handler);
    2. 消息队列:MessageQueue</br>
      以队列的形式对外提供插入和删除的工作,其内部结构是以链表的形式存储消息的。
    3. Looper</br>
      Looper是循环的意思,它负责从消息队列中循环的取出消息然后把消息交给目标(Handler)处理。
    4. Handler</br>
      消息的真正处理者,具备获取消息、发送消息、处理消息、移除消息等功能。
    5. 线程</br>
      线程,CPU调度资源的基本单位。Android中的消息机制也是基于线程中的概念。
    6. ThreadLocal</br>
      可以理解为ThreadLocalData,ThreadLocal的作用是提供线程内的局部变量(TLS),这种变量在线程的生命周期内起作用,每一个线程有他自己所属的值(线程隔离)。

    为什么需要消息机制

    不阻塞UI线程

    Android应用程序启动时,系统会创建一个主线程,负责与UI组件(widget、view)进行交互,比如控制UI界面界面显示、更新等;分发事件给UI界面处理,比如按键事件、触摸事件、屏幕绘图事件等,因此,Android主线程也称为UI线程。
    由此可知,UI线程只能处理一些简单的、短暂的操作,如果要执行繁重的任务或者耗时很长的操作,比如访问网络、数据库、下载等,这种单线程模型会导致线程运行性能大大降低,甚至阻塞UI线程,如果被阻塞超过5秒,系统会提示应用程序无相应对话框,缩写为ANR,导致退出整个应用程序或者短暂杀死应用程序。

    并发程序设计的有序性

    单线程模型的UI主线程也是不安全的。线程安全简单理解为:当一个线程访问功能资源时,对该资源进程了保护,比如加了锁机制,当前线程在没有访问结束释放锁之前,其他线程只能等待直到释放锁才能访问,这样的线程就是安全的。

    Android只允许主线程更新UI界面。子线程处理后的结果无法和主线程交互,即无法直接访问主线程,这就要用到Handler机制来解决此问题。基于Handler机制,在子线程先获得Handler对象,该对象将数据发送到主线程消息队列,主线程通过Loop循环获取消息交给Handler处理。

    消息机制架构

    我们平时接触到的Looper、Message、Handler都是用JAVA实现的,Android是一个基于Linux的系统,底层用C、C++实现的,而且还有NDK的存在,Android消息驱动的模型为了消息的及时性、高效性,在Native层也设计了Java层对应的类如Looper、MessageQueue等。


    Android消息机制架构

    运行机制

    运行机制

    运行机制一句话总结为:Looper不断从MessageQueue中取出一个Message,然后交给其对应的Handler处理。

    总结

    1. Looper依赖于MessageQueue和Thread,每个Thread只对应一个Looper,每个Looper只对应一个MessageQueue(一对一)。
    2. MessageQueue依赖于Message,每个MessageQueue中有N个待处理消息(一对N)。
    3. Message依赖于Handler来进行处理,每个Message有且仅有一个对应的Handler。(一对一)
    4. Handler中持有Looper和MessageQueue的引用,可直接对其进行操作。

    普通的线程是没有Looper的,如果需要Looper对象,那么必须要先调用Looper.prepare()方法,而且一个线程只能有一个Looper。调用完以后,此线程就成为了所谓的LooperThread,若在当前LooperThread中创建Handler对象,那么此Handler会自动关联到当前线程的Looper对象,也就是拥有Looper的引用。

    用法如下:

    class LooperThread extends Thread {
        public Handler mHandler;
        
        public void run() {
            Looper.prepare();
            mHandler = new Handler() {
            public void handleMessage(Message msg) {
            // process incoming messages here
            }
        };
        
        Looper.loop();
    }
    

    Looper是通过管道(pipe)实现的。关于管道,简单来说,管道就是一个文件。在管道的两端,分别是两个打开文件文件描述符,这两个打开文件描述符都是对应同一个文件,其中一个是用来读的,别一个是用来写的。

    相关文章

      网友评论

        本文标题:Android消息机制

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