zoukankan      html  css  js  c++  java
  • Android消息传递之Handler消息机制

    前言:

          无论是现在所做的项目还是以前的项目中,都会遇见线程之间通信、组件之间通信,目前统一采用EventBus来做处理,在总结学习EventBus之前,觉得还是需要学习总结一下最初的实现方式,也算是不忘初心吧,这也是今天来学习总结Handler消息机制的一个原因。

      消息传递相关文章地址:

    Handler机制产生背景

          一个Android应用程序被创建的时候都会创建一个UI主线程,但是有时我们会有一些比较耗时的操作,为了防止阻塞UI主线程,我们会将耗时的操作放到子线程中进行处理,处理完之后操作UI,但是Android不允许子线程操作UI,违背了Android单线程模型的原则(即 Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行),所以Android通过Handler消息机制来实现线程之间的通讯。

    Handler机制主要角色

        Message:消息,其中包含了消息ID,消息处理对象以及处理的数据等,由MessageQueue统一列队,终由Handler处理。 

        Handler:处理者,负责Message的发送及处理。使用Handler时,需要实现handleMessage(Message msg)方法来对特定的Message进行处理,例如更新UI等。 

        MessageQueue:消息队列,用来存放Handler发送过来的消息,并按照FIFO规则执行。当然,存放Message并非实际意义的保存,而是将Message以链表的方式串联起来的,等待Looper的抽取。 

        Looper:消息泵,不断地从MessageQueue中抽取Message执行。因此,一个MessageQueue需要一个Looper。 

        Thread:线程,负责调度整个消息循环,即消息循环的执行场所。

    Handler机制主要运用

        1.)发送消息,在不同的线程间发送消息,使用的方法为sendXXX();
    sendEmptyMessage(int);//发送一个空的消息
    sendMessage(Message);//发送消息,消息中可以携带参数
    sendMessageAtTime(Message, long);//未来某一时间点发送消息
    sendMessageDelayed(Message, long);//延时Nms发送消息

     举例:

     主线程定义Handler

      Handler mHandler = new Handler() {
    
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                switch (msg.what) {
                    case 0:
                        //完成主界面更新,拿到数据
                        String data = (String) msg.obj;
                        textView.setText(data);
                        break;
                    default:
                        break;
                }
            }
        };

    子线程发消息,通知Handler完成UI更新

        private void getDataFromNet() {
            new Thread(new Runnable() {
    
                @Override
                public void run() {
                    //耗时操作,完成之后发送消息给Handler,完成UI更新;
                    mHandler.sendEmptyMessage(0);
    
                    //需要数据传递,用下面方法;
                    Message msg = new Message();
                    msg.obj = "网络数据";//可以是基本类型,可以是对象,可以是List、map等;
                    mHandler.sendMessage(msg);
                }
    
            }).start();
        }
    2.)计划任务,在未来执行某任务,使用的方法为postXXX();
    post(Runnable);//提交计划任务马上执行
    postAtTime(Runnable, long);//提交计划任务在未来的时间点执行
    postDelayed(Runnable, long);//提交计划任务延时Nms执行

    示例:

    主线程定义Handler

      private Handler mHandler=new Handler();

    子线程提交任务更新UI

        private void getDataFromNet() {
            new Thread(new Runnable() {
    
                @Override
                public void run() {
                    //耗时操作,完成之后提交任务更新UI
                    final String data = "网络数据";
                    mHandler.post(new Runnable() {
                        @Override
                        public void run() {
                            textView.setText(data);
                        }
                    });
                }
            }).start();
        }

    Handler机制扩展

        为了更加方便的使用Handler消息机制,Android也提供了几种扩展方式,内部实现都是基于Handler消息机制

      1.) Activity.runOnUiThread(Runnable)
        private void getDataFromNet() {
            new Thread(new Runnable() {
    
                @Override
                public void run() {
                    //耗时操作,完成之后提交任务更新UI
                    final String data = "网络数据";
                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            textView.setText(data);
                        }
                    });
                }
            }).start();
        }
    2 .)View.post(Runnable)
       private void getDataFromNet() {
            new Thread(new Runnable() {
    
                @Override
                public void run() {
                    //耗时操作,完成之后提交任务更新UI
                    final String data = "网络数据";
                    textView.post(new Runnable() {
                        @Override
                        public void run() {
                            textView.setText(data);
                        }
                    });
                }
            }).start();
        }

    5.)使用AsyncTask代替Thread

        private void getDataFromNet() {
            MyTask task = new MyTask();
            task.execute();
        }
    
        private class MyTask extends AsyncTask {
    
            //后台线程执行时
            @Override
            protected Object doInBackground(Object... params) {
                ////耗时操作,
                String data = "网络数据";
                return data;
            }
    
            //后台线程执行结束后的操作,其中参数result为doInBackground返回的结果
            @Override
            protected void onPostExecute(Object result) {
                super.onPostExecute(result);
                textView.setText((String) result);
            }
        }

    小结:

        至此Handler消息简单的总结到此结束,更多内部具体实现可以参考

       1.http://www.cnblogs.com/angeldevil/p/3340644.html

       2.http://blog.csdn.net/fei0724/article/details/8664462

  • 相关阅读:
    log4net 配置 一站式解决
    设计模式-职责链模式(ChainOfResponsibility)
    springboot+Kafka(生产者和消费者)
    springboot启动过程分析
    Eureka注册中心原理
    JDK8常量池整理
    第2章 Java内存区域与内存溢出异常
    第5章 数据库分库分表实例
    物理分页和内存分页-引用
    spring4体系架构
  • 原文地址:https://www.cnblogs.com/whoislcj/p/5590615.html
Copyright © 2011-2022 走看看