1. Android消息处理机制: Handler, MessageQueue, Looper, Thread

线程概念 : 一个应用程序运行时它的主体被称为进程,
一个进程内部可以有多个线程,
线程共享进程的资源 
线程间通信

在android系统中是怎么封装通讯的,假如存在两个线程A和B,如果A线程要告诉B线程一些消息,怎么实现?

A进程怎么发发消息?(1)构造消息,消息里面有数据信息和处理函数(2)发消息;这两个步骤在android源码中被封装成Handler

消息在android中被封装为Message,A发送Message给B,可能B处理不过来,因此A把消息放到B进程里面的消息队列MessageQueue类,在B进程的循环体中做什么事?

(1)、从队列中取出消息;(2)处理消息,执行消息的处理函数;这两个步骤在android源码中被封装成Looper

Looper源码在frameworks/base/core/java/android/os/Looper.java

Handler源码在frameworks/base/core/java/android/os/Handler.java   //ctrl+shift+n在输入handler.java来打开

a. 创建MessageQueue: Looper.prepare()
b. 使用Handler构造、发送Message
b.1 new Handler  //创建handler的时候可以指定Looper即消息接受者和Callback回调函数即消息的处理函数,如果不知道肯定有默认的Looper和消息处理函数
b.2 Handler.sendMessage, sendEmptyMessageAtTime, sendMessageDelayed

c. 使用Looper循环处理消息:在loop函数中有个for死循环
c.1 从MessageQueue中取出Message,//Message msg = queue.next();
c.2 执行它的处理函数: msg.target.dispatchMessage(msg)//target就是Handler,dispatchMessage会调用处理函数

  

应用程序编写:

功能说明:app中创建一个button,创建子线程,添加消息处理功能,并且给Button添加处理函数,并且主线程会监测button,当button按下时给子线程发送消息,子线程收到消息后打印出来

import android.view.View;

import android.util.Log;//打印导入的库

public class MainActivity extends AppCompatActivity{

  private Button mButton;

  private final String TAG = "MessageTest";

  private int ButtonCount = 0;

  private Thread myThread;

  private MyThread myThread2;

  private Handler mHandler;

  private int mMessageCount = 0;

  class MyRunnable:implements Runable{

    int count=0;

    public void run(){

      for(;;){

        Log.d(TAG,"MyThread"+(count++));

        try{  

          Thread.sleep(3000);

        }catch(INterrupteException e){

          e.printStackTrace();

        }

      }

    }

  }

  class Mythread extends Thread{       //android只带的消息处理是在HandlerThread.java中,其功能同我们怎么的Mythread,如果不自己创建Mythread,可以直接使用HandlerThread类

    public void run(){

      super.run();

      Looper.prepare();//创建消息队列

      Looper.loop();//从消息队列中取出消息,调用消息的处理函数

    }

    public Looper getLooper(){

      return Looper.myLooper();

    }

  }

  pretected void onCreate(Bundle savedInstanceState){

    ........

    ........

    mButton = (Button)findViewById(R.id.button);//双击Button,按下shift+F1,可以查看类的帮助文档

    mButton.setOnClickListener(new View.OnClickListener(){

      public void onClick(View v){

        Log.d(TAG,"SendMessage"+(ButtonCount++) );

        Message msg = new Message();

        mHandler.sendMessage(msg );//发送消息的会从把消息放到消息队列中,消息队列是从handler创建的时候从Looper中获得

      }

    });

    myThread = new Thread(new MyRunnable,"MessageTestThread");//(runnable接口的方法run是线程的主体函数)

    myThread.start();

    myThread2 = new MyThread();

    myThread2 .start();//功能和myThread一样,start会导致MyThread的run函数被执行,run函数会执行Looper.prepare()去创建消息队列,但这个函数并不一定马上就执行,如果住进程中创建Handler的时候没有队列,会存在风险,所以修改class MyThread extends Thread{}

    ////我们现在的代码里面有三个线程了,主线程、myThread和myThread2,,要确定消息发给谁Looper

    mHandler = new Handler(myThread2.getLooper(),new Handler.Callback(){

      public boolean handleMessage(Message msg){

        Log.d(TAG,"getMessage"+(mMessageCount++))

        return false;

      }

    });

  }

}

针对上面所说的风险修改class MyThread extends Thread{},如下:

class MyThread extends Thread{

  private Looper mLooper;

    public void run(){

      super.run();

      Looper.prepare();//创建消息队列

      mLooper = Looper.myLooper();

      notifyAll();//当主线程调用getLooper,在mLooper 为空的时候会休眠,所以这里需要唤醒休眠的线程

      Looper.loop();//从消息队列中取出消息,调用消息的处理函数

    }

    public Looper getLooper(){

      if(!isAlive()){

        return null;

      }

      synchronized(this){

        while(isAlive() && mLooper == null){

          try{

            wait();

          }catch (InterruptedException e){

          }

        }

      }

      return mLooper;

    }

}

修改代码使用系统只带的消息处理线程:

1、导入包:import android.os.HandlerThread;

2、在public class MainActivity extends AppCompatActivity中新增

  private HandlerThread myThread3;

  private Handler mHandler3;

  myThread3 = new HandlerThread("MessageTestThread3");

  myThread3 .start();

  mHandler3 = new Handler(myThread3.getLooper());

  在button的onClick函数中添加:

  mHandler3.post(new Runnable(){

    public void run(){

      Log.d(TAG,"getMessage for Thread3"+(mMessageCount++));

    }

  });

  

7.1 基础知识Android消息处理机制的更多相关文章

  1. 【Android 开发】: Android 消息处理机制之一: Handler 与 Message

    最近几讲内容,我们学习了Android中关于多线程的一些知识,上一讲我们讲解了异步任务 AsyncTask 的操作,Android中还提供了其他的线程操作,如Handler Message Messa ...

  2. (转)Android消息处理机制(Handler、Looper、MessageQueue与Message)

    转自 http://www.cnblogs.com/angeldevil/p/3340644.html Android消息处理机制(Handler.Looper.MessageQueue与Messag ...

  3. Android消息处理机制

    Android消息处理机制 Android应用程序消息处理机制(深入到native,实际由管道实现-pipe&epoll)

  4. 解析Android消息处理机制:Handler/Thread/Looper & MessageQueue

    解析Android消息处理机制 ——Handler/Thread/Looper & MessageQueue Keywords: Android Message HandlerThread L ...

  5. Android架构分析之Android消息处理机制(二)

    作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz Android版本号:4.4.2 在上一篇文章中我们看了一个使用Handler处理Message消息的样例,本文我们 ...

  6. Android架构分析之Android消息处理机制(一)

    作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz Android版本号:4.4.2 在这个系列文章中我们将来分析Android消息处理机制. 本文介绍了一个使用Han ...

  7. Android消息处理机制(Handler 与Message)---01

    一.handler的使用场景为么会有handler?(部分内容图片摘自http://www.runoob.com/w3cnote/android-tutorial-handler-message.ht ...

  8. [旧][Android] 消息处理机制

    备注 原发表于2016.06.06,资料已过时,仅作备份,谨慎参考 概述 Android 的消息处理机制主要是指 Handler 的运行机制以及 Handler 所附带的 MessageQueue 和 ...

  9. 【Android】Android消息处理机制

    三大核心类 android的消息处理有三个核心类:Looper,Handler和Message. 其实还有一个Message Queue(消息队列),但是MQ被封装到Looper里面了 Looper ...

随机推荐

  1. Android 学习笔记进阶十二之裁截图片

    package xiaosi.cut; import java.io.File; import android.app.Activity; import android.content.Intent; ...

  2. js---对象 和 函数this

    一:对象创建的方法 //普通 字面量形式 var obj = { name:'名字', fn:function(){ console.log(this.name); } } //new 实例 var ...

  3. 71.lambda表达式的递归

    #include <iostream> #include <functional> using namespace std; void main() { //&调用外部 ...

  4. Kinect 开发 —— 用户交互设计的若干思考

    Metro 风格 windows 8 Kinect Hub 手势原型设计 悬停选择     翻页控制 关节点重叠的处理方法 将箭靶设置在画面的边缘,这样玩家持弓的角度与屏幕保持一个大约45度的锐角,这 ...

  5. 搭建Spark源码研读和代码调试的开发环境

    转载自https://github.com/linbojin/spark-notes/blob/master/ide-setup.md 搭建Spark源码研读和代码调试的开发环境 工欲善其事,必先利其 ...

  6. C_数组详解

    数组: 一 一维数组 1.1 一维数组的定义: 类型符 数组名[常量表达式]; int a[10]; 说明: 1.数组的命名规则遵循标识符命名规则. 2.定义时需要指定元素的个数.方括号里的常量表达式 ...

  7. vue 使用同一组件,切换时不触发created、mounted钩子

    两个页面参数不同使用同一组件,默认情况下当这两个页面切换时并不会触发created或者mounted钩子. 方法一:通过watch $route的变化来做处理 watch: { $route() { ...

  8. linux系统常用日志

    系统日志记录着系统运行中的记录信息,在服务或者系统发生故障的时候,通过查询系统日志,可以帮助我们诊断.系统日志可以预警安全问题,系统日志一般都存放在/var/log目录下 /var/log/dmesg ...

  9. Python 3 与"Hello World!"

    Python 3 版本 Python的3.0版本,常被称为Python 3000,或简称Py3k.相对于Python的早期版本,这是一个较大的升级.为了不带入过多的累赘,Python 3.0在设计的时 ...

  10. 【Hello 2018 B】Christmas Spruce

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 写个dfs看看是不是每个节点都有3个叶子节点就可以了. [代码] #include <bits/stdc++.h> us ...