说到Handler想必大家都经常用到,在非UI线程更新UI那可是利器,用起来也非常容易上手

从使用上来说,我们只需要关注sendMessage和handleMessage即可

所以我们先从Handler和Message来说起,先看一小段代码

    public static final int UPDATE_TEXT_VIEW = 0;
public TextView mResultTextView = null; // new 一个 Handler 对象, 以内部类的方式重写 handleMessage 这个函数
public Handler mMyHandler = new Handler() {
// ③ 处理消息
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case UPDATE_TEXT_VIEW:
/* 这里可更新 ui */
mResultTextView.setText(String.valueOf(msg.arg1));
break; default:
/* do nothing */
break;
}
};
}; // ==================================================================
// 函数名: calc
// 日期: 2015-08-30
// 功能: 计算入参并显示在 UI 上,然后后续以每秒 +1 在 UI 上更新
// 输入参数: int a
// int b
// 返回值: 无
// 修改记录:千里草新增函数
// ==================================================================
public void calc(int a, int b) {
int c = a + b;
final Message msg = new Message();
msg.what = UPDATE_TEXT_VIEW;
msg.arg1 = c;
// ① 发送消息
mMyHandler.sendMessage(msg);
// 这里更新 UI 线程的TextView OK
// mResultTextView.setText(String.valueOf(msg.arg1)); // 启动一个新的线程来每秒刷新 TextView
new Thread() {
public void run() {
while (true) {
msg.arg1++;
// ② 发送消息
mMyHandler.sendMessage(msg); // mResultTextView.setText(String.valueOf(msg.arg1)); 这里会报错
try {
sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} };
}.start();
}

代码中有①,②两处发送消息,. 分别是从UI线程和从子线程发送的消息, 在③处两个消息都可以被准确的接收到.

在接受消息函数handleMessage 里我们可以做任何想做的事情,包括更新UI…

从使用来说,大家根据这个例子,想必已经可以简单使用Handler来处理多线程之间的交互了…

此处需要注意的是在使用Handler时,必须需要重写handleMessage 才能达到我们想达到的目的...(从逻辑上来说,没有接受消息的地方,发送的消息有何意义呢,是吧….从原理上..我们接下来会谈到……)

谈到Handler,我们还可以用它执行一个Runnable多线程…例如下面的代码..

mMyHandler.post(new Runnable() {

            @Override
public void run() {
// 这里需要注意的是,calc 是在子线程里被调用,所以calc 就无法操作 UI 了.
calc(1, 2);
}
});

Handler可真神奇,既可以发消息也可以执行某个任务线程.

为了更深层次的了解它,只能去看看它的源码了.let’s go!

先看看是如何post一个Runnable线程的

public final boolean post(Runnable r)
{
//这里实际上是发送了一个Message
return sendMessageDelayed(getPostMessage(r), 0);
} //组装一个只带Runnable的Message
private static Message getPostMessage(Runnable r) {
Message m = Message.obtain();
m.callback = r;
return m;
} //发送消息
public final boolean sendMessageDelayed(Message msg, long delayMillis)
{
if (delayMillis < 0) {
delayMillis = 0;
}
return sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis);
}

哎哟喂,这Handler小丫头片子,还假装两幅面孔呢, 经查看源码,发现它实际上是发送了一个Message消息,然而这个消息仅仅携带了一个Runnable对象.

现在来看只需要分析Handler和Message了..我们来看看最终Handler是如何发送消息的..我们继续看源码,非得把他们扒个精光不可

public boolean sendMessageAtTime(Message msg, long uptimeMillis)
{
boolean sent = false;
//消息队列,这里存储Handler发送的消息
MessageQueue queue = mQueue;
if (queue != null) {
//Message与Handler绑定
msg.target = this;
//消息进队列
sent = queue.enqueueMessage(msg, uptimeMillis);
}
else {
RuntimeException e = new RuntimeException(
this + " sendMessageAtTime() called with no mQueue");
Log.w("Looper", e.getMessage(), e);
}
return sent;
}

从消息最终被发出的函数来看,这里只做了两个事情:

1.Message与Handler绑定

2.将Message放进消息队列里

消息进队列之后,Handler发送消息的任务算是完美的完成了,接下来我们该介绍介绍Looper了,如果没有Looper,Handler和Message之间的爱情可以说是不完整的.

我们先来看看一个Looper是如何使用的吧

class LooperThread extends Thread {
public Handler mHandler; public void run() {
// 为当前线程准备一个 Looper
Looper.prepare(); mHandler = new Handler() {
public void handleMessage(Message msg) {
// 处理消息
}
};
// Looper 开始工作
Looper.loop();
}
}
    //继续看源码,prepare执行之后sThreadLocal绑定一个Looper对象
//sThreadLocal 能且只能绑定一个Looper对象
public static void prepare() {
prepare(true);
} private static void prepare(boolean quitAllowed) {
if (sThreadLocal.get() != null) {
throw new RuntimeException("Only one Looper may be created per thread");
}
sThreadLocal.set(new Looper(quitAllowed));
} /**
* Run the message queue in this thread. Be sure to call
* {@link #quit()} to end the loop.
*/
public static void loop() {
final Looper me = myLooper();
if (me == null) {
throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
}
final MessageQueue queue = me.mQueue; // Make sure the identity of this thread is that of the local process,
// and keep track of what that identity token actually is.
Binder.clearCallingIdentity();
final long ident = Binder.clearCallingIdentity(); //无条件的for循环,来遍历Messagequeue
for (;;) {
//获取下一条消息
Message msg = queue.next(); // might block
if (msg == null) {
// No message indicates that the message queue is quitting.
return;
} // This must be in a local variable, in case a UI event sets the logger
Printer logging = me.mLogging;
if (logging != null) {
logging.println(">>>>> Dispatching to " + msg.target + " " +
msg.callback + ": " + msg.what);
} //Message绑定的Handler来开始处理消息.. 下面贴上dispatchMessage,函数,大家继续往下看
msg.target.dispatchMessage(msg); if (logging != null) {
logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);
} // Make sure that during the course of dispatching the
// identity of the thread wasn't corrupted.
final long newIdent = Binder.clearCallingIdentity();
if (ident != newIdent) {
Log.wtf(TAG, "Thread identity changed from 0x"
+ Long.toHexString(ident) + " to 0x"
+ Long.toHexString(newIdent) + " while dispatching to "
+ msg.target.getClass().getName() + " "
+ msg.callback + " what=" + msg.what);
} msg.recycle();
}
}
/**
* Callback interface you can use when instantiating a Handler to avoid
* having to implement your own subclass of Handler.
*/
public interface Callback {
public boolean handleMessage(Message msg);
} /**
* Subclasses must implement this to receive messages.
* 子类必须完成handleMessage这个函数来接收消息
*/
public void handleMessage(Message msg) {
} /**
* Handle system messages here.
*/
public void dispatchMessage(Message msg) {
//收到消息后,先处理Runnable 对象callback.
if (msg.callback != null) {
handleCallback(msg);
} else {
//如果这里有回调接口,那么就直接调用该接口,不再继续调用Handler的handleMessage函数来处理消息
if (mCallback != null) {
if (mCallback.handleMessage(msg)) {
return;
}
}
handleMessage(msg);
}
}

从上述加了中文注释的代码中能看出,Handler发送的Message通过Looper分发给了各自的Handler.

所以上面说的在使用Handler时,必须需要重写handleMessage 才能达到我们想达到的目的...也不完全正确,

我们也可以实现一个CallBack接口来处理消息,用Google的原话是

    /**
* Callback interface you can use when instantiating a Handler to avoid
* having to implement your own subclass of Handler.
*/
public interface Callback {
public boolean handleMessage(Message msg);
}

再贴最后一段代码来彻彻底底说明Handler Looper之间的关系, 那就是Handler的 构造函数

/**
* Default constructor associates this handler with the queue for the
* current thread.
*
* If there isn't one, this handler won't be able to receive messages.
*/
public Handler() {
if (FIND_POTENTIAL_LEAKS) {
final Class<? extends Handler> klass = getClass();
if ((klass.isAnonymousClass() || klass.isMemberClass() || klass.isLocalClass()) &&
(klass.getModifiers() & Modifier.STATIC) == 0) {
Log.w(TAG, "The following Handler class should be static or leaks might occur: " +
klass.getCanonicalName());
}
}
//绑定当前线程的Looper
mLooper = Looper.myLooper();
if (mLooper == null) {
throw new RuntimeException(
"Can't create handler inside thread that has not called Looper.prepare()");
}
//与当前线程的消息队列绑定
mQueue = mLooper.mQueue;
mCallback = null;
}

一句话总结来说,Handler初始化时绑定了线程的MessageQueue,当前线程的Looper来依次分发MessageQueue里的消息.

MessageQueue的消息会根据Message的target绑定的Handler被Looper分发到各自的Handler里去处理.

例如Activity里有两个HandlerA和HandlerB, HandlerA发送的消息绝对不可能被HandlerB处理..

先暂时写到这里吧,后续补上流程图和类图

上述只是个人的很片面的理解,希望大家补充和指出不足的地方…

2015年9月4日 04:10:05 千里草

Handler和Message以及Looper之间的三角关系的更多相关文章

  1. Android的消息处理机制,handler,message,looper(一)

    当应用程序启动时,Android首先会开启一个主线程(也就是UI线程),主线程为管理界面中的UI控件.在程序开发时,对于比较耗时的操作,通常会为其开辟一个单独的线程来执行,以尽可能减少用户的等待时间. ...

  2. Android 消息机制 (Handler、Message、Looper)

    综合:http://blog.csdn.net/dadoneo/article/details/7667726 与 http://android.tgbus.com/Android/androidne ...

  3. Android面试题 请解释下单线程模型中Message、Handler、MessageQueue、Looper之间的关系

    简单的说,Handler获取当前线程中的looper对象,looper用来存放从MessageQueue中取出的Message,再由Handler进行Message分发和处理,按照先进先出执行. Me ...

  4. Android开发学习之路--异步消息Handler,Message,Looper和AsyncTask之初体验

    在简易音乐播放器中,用了Handler,也没有过多地去研究学习,这里再学习下android下的异步消息处理机制.这里用了Handler主要是在线程中不能更新UI,而需要通过Handler才可以.关于异 ...

  5. Handler,Thread,Looper之间关系小结

    http://blog.csdn.net/sunxingzhesunjinbiao/article/details/6794840 (1) Looper类别用来为一个线程开启一个消息循环.默认情况下A ...

  6. javascript中prototype、constructor以及__proto__之间的三角关系

    三者暧昧关系简单整理 在javascript中,prototype.constructor以及__proto__之间有着“著名”的剪不断理还乱的三角关系,楼主就着自己对它们的浅显认识,来粗略地理理以备 ...

  7. 请解释下在单线程模型中Message、Handler、MessageQueue、Looper之间的关系

    对于面试,每个职场人士都经历过,面试官更看中你对于技术的理解是否透彻,需要知其所以然,而实际工作中看中的工作效率,都是在使用API的角度来完成任务,当在一家公司呆久了有跳槽的想法时,个人的亲身经历就是 ...

  8. Looper Handler MessageQueue Message 探究

    Android消息处理的大致的原理如下: 1.有一个消息队列,可以往队列中添加消息 2.有一个消息循环,可以从消息队列中取出消息 Android系统中这些工作主要由Looper和Handler两个类来 ...

  9. 深入了解Looper、Handler、Message之间关系

    深入了解Looper.Handler.Message之间关系 前言及简介 上个星期我们整个项目组趁着小假期,驱车去了江门市的台山猛虎峡玩了两个多钟左右极限勇士全程漂流,感觉真得不错,夏天就应该多多玩水 ...

随机推荐

  1. Spring Boot 系列教程17-Cache-缓存

    缓存 缓存就是数据交换的缓冲区(称作Cache),当某一硬件要读取数据时,会首先从缓存中查找需要的数据,如果找到了则直接执行,找不到的话则从内存中找.由于缓存的运行速度比内存快得多,故缓存的作用就是帮 ...

  2. hiho1246(数学求模)

    input 1<=n<=2000 a1 a2 ... an 1<=ai<=5*10e7 output n行,第i行指切成i段,每段和的最大公约数的最大值 做法:环形数组切成n段 ...

  3. 在DLL中导出另一静态库中的函数

    开发环境: win7_x64.VS2013 应用场景: 动态库A依赖动态库B,而动态库B又使用了静态库C:有些情况下,我们需要将C从B里面导出,然后提供给A使用. 正文: Step1: 1.新建测试静 ...

  4. Android图片处理神器BitmapFun源码分析

    作为一名Android开发人员,相信大家对图片OOM的问题已经耳熟能详了,关于图片缓存和解决OOM的开源项目也是相当的多,被大家熟知的就是Universal_image_loader和Volley了, ...

  5. myeclipse中常用的快捷键

    存盘 Ctrl+s(肯定知道) 注释代码 Ctrl+/ 取消注释 Ctrl+\(Eclipse3已经都合并到Ctrl+/了) 代码辅助 Alt+/ 快速修复 Ctrl+1 代码格式化 Ctrl+Shi ...

  6. android 5.0新特性学习--RecyclerView

    在过去很多年,我们的PC或者手机设备都是采用拟物化的设计风格,IOS采用扁平化的特性,android在2014年IO大会上说采用Material Design的设计风格,显示效果不能过于生硬的转换,而 ...

  7. PAT1015

    A reversible prime in any number system is a prime whose "reverse" in that number system i ...

  8. 代码异味---Code smell

    程序员应该竭尽全力去写那些重复的代码.以下几点是我目前最需要改进的地方. 重复代码: 相同或者相似的代码存在于一个以上的地方. 长方法: 一个非常长的方法.函数或者过程. 巨类: 一个非常庞大的类. ...

  9. svn log操作

    查看当前文件夹的最近N次提交记录 svn update; svn log --limit <N> -v 含义是:查询最近N次提交记录的详细信息,包括版本号,提交文件列表,log信息 对比某 ...

  10. AVR之BOOTLOADER技术详解(转)

    源:http://blog.csdn.net/zhenhua10/article/details/6442412 ATmega128具备引导加载支持的用户程序自编程功能(In-System Progr ...