一、概念

1、事件驱动型

什么是事件驱动?就是有事了才去处理,没事就躺着不动。假如把用户点击按钮,滑动页面等这些都看作事件,事件产生后程序就执行相应的处理方法,就是属于事件驱动型。

2、消息循环

把需要处理的事件表示成一个消息,并且把这个消息放入一个队列。消息循环就是一循环,for或者while都一样。从消息队列里面取出未处理的消息,然后调用该消息的处理方法。

3、Handler

最开始碰到handler是需要在子线程里面更新UI的时候。android的UI更新只能在主线程中进行,但是在子线程中执行的逻辑又需要更新UI,例如文件下载,在子线程中访问网络下载之后,就是更新下载进度。这个时候就需要使用Hanlder,准确的说是要发送一个进度更新的消息。什么是Handler?我的理解是消息的处理者。create消息对应一个create的Handler,destroy消息对应一个destroy的Handler。

二、实现

只是说说概念太假了,下面就来实现一个简单的消息处理机制。

1、Msg

把产生的事件用消息来表示,数据用各个参数传递。

public class Msg implements Serializable{
//序列化标识
private static final long serialVersionUID = -2414053244664115328L; //该消息的处理者。
private int handlerId; //参数。
public Object arg1;
public Object arg2; //大量参数
public Object array[]; public Msg(int handlerId) {
this.handlerId=handlerId;
} public void setHandlerId(int handlerId) {
this.handlerId=handlerId;
}
public int getHandleId() {
return handlerId;
}
}

2、Handler

事件的处理者

public abstract class Handler {
//唯一标识,由Looper分配
private int id; //使用该Handler的Looper
private Looper looper; public Handler(Looper looper) {
this.looper=looper;
id=looper.addMsgHandler(this);
} //消息处理函数
abstract public boolean handleMsg(Msg msg); //添加一个未处理消息。
public void sendMsg(Msg msg) {
looper.addMsg(msg);
} //返回该handler的信使。
public Msg obtainMsg() {
return new Msg(id);
}
}

3、Looper

消息循环

import java.util.HashMap;
import java.util.LinkedList;
import java.util.Map;
import java.util.Queue; public class Looper {
private int handlerCount=0;
//消息队列。
private Queue<Msg> msgQueue=new LinkedList<Msg>();
//消息的处理。
private Map<Integer, Handler> msgHandler=new HashMap<>(); //loop
public void loop() {
for(;true;)
if(!msgQueue.isEmpty())
if(!distributeMsg(msgQueue.poll()))
//当消息处理返回false时。程序结束。
break;
} //添加处理消息的handler
public int addMsgHandler(Handler handler) {
handlerCount++;
msgHandler.put(handlerCount,handler);
return handlerCount;
} //添加待处理的消息
public void addMsg(Msg msg) {
msgQueue.add(msg);
} //消息分发
private boolean distributeMsg(Msg msg) {
Handler handler=msgHandler.get(msg.getHandleId());
if(handler!=null) {
handler.handleMsg(msg);
}else {
//出现未知消息,程序结束。
System.out.println("exit");
return false;
}
return true;
}
}

4、模拟生命周期

abstract class Basic{
private Looper mainLooper=new Looper();
private Handler sysHandler=new Handler(mainLooper) {
@Override
public boolean handleMsg(Msg msg) {
if(msg.arg1.equals("create")) {
onCreate();
}
if(msg.arg1.equals("destroy")) {
onDestroy();
}
return true;
}
}; public Basic() {
Msg m=sysHandler.obtainMsg();
m.arg1="create";
sysHandler.sendMsg(m);
//新获取一个Msg,不能沿用上一个。
m=sysHandler.obtainMsg();
m.arg1="destroy";
sysHandler.sendMsg(m);
} public Looper getMainLooper() {
return mainLooper;
} /**
* 生命周期
*/
abstract public void onCreate();
abstract public void onDestroy();
}

上面的代码创建了一个抽象类Basic,在里面注册了一个处理create和destroy两个消息的Handler。

5、子线程调用主线程的方法。

public class Main extends Basic{
private final Handler handler
=new Handler(getMainLooper()) {
@Override
public boolean handleMsg(Msg msg) {
// TODO Auto-generated method stub
System.out.println("msg.arg1="+msg.arg1+",Tid="+
Thread.currentThread().getId());
getMainLooper().addMsg(new Msg(-1));
return true;
}
};
public void onCreate() {
System.out.println("..............onCreate");
System.out.println("mainThread Tid="+Thread.currentThread().getId());
new Thread(new Runnable() {
@Override
public void run() {
// TODO Auto-generated method stub
System.out.println("childThread Tid="+Thread.currentThread().getId());
Msg msg=handler.obtainMsg();
msg.arg1="childCall";
handler.sendMsg(msg);
}
}).start();
}
@Override
public void onDestroy() {
// TODO Auto-generated method stub
System.out.println(".............onDestroy");
//getMainLooper().addMsg(new Msg(-1));
}
public static void main(String[] args) {
new Main().getMainLooper().loop();
}
}

6、结果

结果分析,首先是两个生命周期的方法被调用,其次是实现了子线程调用主线程的方法。这里子线程转到主线程的原因是因为Looper运行在主线程,消息由Looper分发处理。

  

  

Android的消息循环与Handler机制理解的更多相关文章

  1. Android源码分析笔记--Handler机制

    #Handler机制# Handler机制实际就是实现一个 异步消息循环处理器 Handler的真正意义: 异步处理 Handler机制的整体表述: 消息处理线程: 在Handler机制中,异步消息处 ...

  2. Android中的Handler机制

    直接在UI线程中开启子线程来更新TextView显示的内容,运行程序我们会发现,如下错 误:android.view.ViewRoot$CalledFromWrongThreadException: ...

  3. Handler机制与生产者消费者模式

    本文梳理了 Handler 的源码,并详细阐述了 Handler 与生产者消费者模式的关系,最后给出了多版自定义 Handler 实现.本文首发于简书,重新整理发布. 一.Handler Handle ...

  4. Android之消息机制Handler,Looper,Message解析

    PS:由于感冒原因,本篇写的有点没有主干,大家凑合看吧.. 学习内容: 1.MessageQueue,Looper,MessageQueue的作用. 2.子线程向主线程中发送消息 3.主线程向子线程中 ...

  5. Android的消息循环机制 Looper Handler类分析

    Android的消息循环机制 Looper Handler类分析 Looper类说明   Looper 类用来为一个线程跑一个消息循环. 线程在默认情况下是没有消息循环与之关联的,Thread类在ru ...

  6. android 进程间通信 messenger 是什么 binder 跟 aidl 区别 intent 进程间 通讯? android 消息机制 进程间 android 进程间 可以用 handler么 messenger 与 handler 机制 messenger 机制 是不是 就是 handler 机制 或 , 是不是就是 消息机制 android messenge

    韩梦飞沙  韩亚飞  313134555@qq.com  yue31313  han_meng_fei_sha messenger 是什么 binder 跟 aidl 区别 intent 进程间 通讯 ...

  7. Android Handler的使用示例:结合源码理解Android Handler机制(一)

    什么是Handler? Android 的官方解释: 文档分节1:A Handler allows you to send and process Message and Runnable objec ...

  8. Android的消息机制: Message/MessageQueue/Handler/Looper

    概览   * Message:消息.消息里面可包含简单数据.Object和Bundle,还可以包含一个Runnable(实际上可看做回调). * MessageQueue:消息队列,供Looper线程 ...

  9. 聊一聊Android的消息机制

    聊一聊Android的消息机制 侯 亮 1概述 在Android平台上,主要用到两种通信机制,即Binder机制和消息机制,前者用于跨进程通信,后者用于进程内部通信. 从技术实现上来说,消息机制还是比 ...

随机推荐

  1. 微信分享(移动web端)

    create-at 2019-02-16 引入微信JS-SDK http://res.wx.qq.com/open/js/jweixin-1.4.0.js (当前最新版本) js 相关代码 (移动端实 ...

  2. 使用sar进行性能分析

    sar可用于监控Linux系统性能,帮助我们分析性能瓶颈.sar工具的使用方式为”sar [选项] intervar [count]”,其中interval为统计信息采样时间,count为采样次数. ...

  3. 体验使用MUI上手练习app页面开发

    因为公司安排需要先学习一点app开发,而安排学习的框架就是MUI,上手两天体验还算可以(来自后端人员的懵逼),靠着MUI的快捷键可以快速的完成自己想要的样式模板,更多的交互性的内容则需要使用js来完成 ...

  4. GoAccess 分析 Nginx 日志

    0x00 事件 帮助朋友搭建了博客,运行过了一段时间,准备发个网站分析报告给他. 有效的数据只有 Nginx 的访问日志,于是使用决定 GoAccess 工具对这个日志进行分析, 0x01 安装 吾使 ...

  5. 盘一盘 NIO (一)—— Buffer源码解析

    Buffer是个啥? Buffer 即缓冲区,用来暂存输入输出数据的区域.Buffer对象是一份固定数量的数据的容器,实质上是一个数组.但是一个缓冲区不仅仅是一个数组,缓冲区提供了对数据的结构化访问, ...

  6. 安装hadoop集群--hdfs

    安装hadoop集群--hdfs 大数据软件 链接:https://pan.baidu.com/s/1-3PYLHMgvvONawJq55hstQ 提取码:izqf 准备一台干净的虚拟机-centos ...

  7. JS函数提升和变量提升

    1.1什么是函数提升和变量的提升? JS引擎在运行整个JS代码的过程中,分为俩步. 第一步是读取和解析JS代码,第二部是执行. 在引擎解析JS代码的时候,当解析器遇见变量声明(var 变量名)和函数声 ...

  8. C#装箱与拆箱总结

    装箱和拆箱是值类型和引用类型之间相互转换是要执行的操作.  1. 装箱在值类型向引用类型转换时发生 2. 拆箱在引用类型向值类型转换时发生 光上述两句话不难理解,但是往深处了解,就需要一些篇幅来解释了 ...

  9. Enum与最佳単例设计

    1 枚举基础 自定义一个枚举类很简单, 不过类型关键字是 enum, 不是 class, 也不是 interface.public enum Action { UP, DOWN, LEFT, RIGH ...

  10. [Environment setup]python+selenium+Eclipse+pydev环境搭建

    转载自: http://www.cnblogs.com/Bonker/p/3584707.html http://www.cnblogs.com/fnng/archive/2013/05/29/310 ...