Android Handler消息传递机制详解

1.为什么要用Handler

  出于性能优化的考虑,Android UI操作并不是线程安全,如果有多个线程并发操作UI组件,可能导致线程安全问题。可以设想下,如果在一个Activity中有多个线程去更新UI,并且都没有加锁机制,可能会导致什么问题? 界面混乱,如果加锁的话可以避免该问题但又会导致性能下降。因此,Android规定只允许UI线程修改Activity的UI组件。当程序第一次启动时,Android会同时启动一条主线程(Main Thread),主线程主要负责处理与UI相关的事件,比如用户按钮事件,并把相关的事件分发到对应的组件进行处理,因此主线程又称为UI线程。那么怎么在新启动的线程中更新UI组件呢,这就需要借助handler的消息传递机制来实现了。

2.Handler简介

  Handler类的主要作用主要有两个:

    1>在新启动的线程中发送消息

    2>在主线程中获取和处理消息

  Handler类包含如下方法用于发送、处理消息。(这里只列出常用的方法,如果想获取更多方法,建议查看api文档:http://developer.android.com/reference/android/os/Handler.html)

    ♦ void handlerMessage(Message msg):处理消息的方法,该方法通常用于被重写。

    ♦ final boolean hasMessage(int what):检查消息队列中是否包含what属性为指定值的消息。

    ♦ sendEmptyMessage(int what):发送空消息

    ♦ final boolean sendMessage(Message msg):立即发送消息,注意这块返回值,如果message成功的被放到message queue里面则返回true,反之,返回false;(个人建议:对于这类问题不必主观去记它,当实际使用时,直接查看源码即可,源码中有详细的注释)

3.Handler、Message、Looper、MessageQueue之间的关系、工作原理

  为了更好的理解Handler,先来看看和Handler相关的一些组件:

    Message:Handler发送、接收和处理的消息对象

    Looper:每个线程只能拥有一个Looper.它的looper()方法负责循环读取MessageQueue中的消息并将读取到的消息交给发送该消息的handler进行处理。

    MessageQueue:消息队列,它采用先进先出的方式来管理Message。程序在创建Looper对象时,会在它的构造器中创建MessageQueue。源码如下:

    

1   private Looper(boolean quitAllowed) {
2         mQueue = new MessageQueue(quitAllowed);
3         mThread = Thread.currentThread();
4     }

    从源码第2行中可以看出,在创建Looper对象时会创建一个与之关联的MessageQueue对象。构造器是private修饰的,所以程序员是无法创建Looper对象的。

    Handler:前面说Handler作用有两个---发送消息和处理消息,Handler发送的消息必须被送到指定的MessageQueue,也就是说,要想Handler正常工作必须在当前线程中有一个MessageQueue,否则消息没法保存。而MessageQueue是由Looper负责管理的,因此要想Handler正常工作,必须在当前线程中有一个Looper对象,这里分为两种情况:

      1>主线程(UI线程),系统已经初始化了一个Looper对象,因此程序直接创建Handler即可

      2>程序员自己创建的子线程,这时,程序员必须创建一个Looper对象,并启动它。

    创建Looper使用:Looper.prepare(),查看源码:

 1    public static void prepare() {
 2         prepare(true);
 3     }
 4
 5   private static void prepare(boolean quitAllowed) {
 6         if (sThreadLocal.get() != null) {
 7             throw new RuntimeException("Only one Looper may be created per thread");
 8         }
 9         sThreadLocal.set(new Looper(quitAllowed));
10     }
11
12   private Looper(boolean quitAllowed) {
13         mQueue = new MessageQueue(quitAllowed);
14         mThread = Thread.currentThread();
15     }

    通过方法调用,第9行创建Looper对象,创建Looper对象时同时会创建MessageQueue对象(第13行)。此外,可以看出prepare()允许一个线程最多有一个Looper被创建。

    然后调用Looper的looper()方法来启动它,looper()使用一个死循环不断取出MessageQueue中的消息,并将消息发送给对应的Handler进行处理。下面是Looper类中looper()方法的部分源码:

 1 for (;;) {
 2             Message msg = queue.next(); // might block
 3             if (msg == null) {
 4                 // No message indicates that the message queue is quitting.
 5                 return;
 6             }
 7
 8             // This must be in a local variable, in case a UI event sets the logger
 9             Printer logging = me.mLogging;
10             if (logging != null) {
11                 logging.println(">>>>> Dispatching to " + msg.target + " " +
12                         msg.callback + ": " + msg.what);
13             }
14
15             msg.target.dispatchMessage(msg);
16
17             if (logging != null) {
18                 logging.println("<<<<< Finished to " + msg.target + " " + msg.callback);
19             }
20
21             // Make sure that during the course of dispatching the
22             // identity of the thread wasn‘t corrupted.
23             final long newIdent = Binder.clearCallingIdentity();
24             if (ident != newIdent) {
25                 Log.wtf(TAG, "Thread identity changed from 0x"
26                         + Long.toHexString(ident) + " to 0x"
27                         + Long.toHexString(newIdent) + " while dispatching to "
28                         + msg.target.getClass().getName() + " "
29                         + msg.callback + " what=" + msg.what);
30             }
31
32             msg.recycleUnchecked();
33         }

    很明显第1行用了一个死循环,第2行从queue中取出Message,第15行通过dispatchMessage(Message msg)方法将消息发送给Handler。

4.HandlerThread介绍

  Android API解释:

  Handy class for starting a new thread that has a looper. The looper can then be used to create handler classes. Note that start() must still be called.

  意思是说:这个类启动一个新的线程并且创建一个Looper,这个Looper可以用来创建一个Handler类,完了之后一定要启动这个线程。

  什么时候使用HandlerThread?

    1.主线程需要通知子线程执行耗时操作(一般都是子线程执行耗时操作,完了之后,发送消息给主线程更新UI)。

    2.开发中可能会多次创建匿名线程,这样可能会消耗更多的系统资源。而HandlerThread自带Looper使他可以通过消息来多次重复使用当前线程,节省开支;

  下面是HandlerThread应用部分代码:

 1   private static final String TAG = "MainActivity";
 2     private static final int FLAG_TEST = 1;
 3
 4     @Override
 5     protected void onCreate(Bundle savedInstanceState) {
 6         super.onCreate(savedInstanceState);
 7         setContentView(R.layout.activity_main);
 8         Log.i(TAG,"main thread:"+Thread.currentThread());
 9         HandlerThread thread = new HandlerThread("handler thread");
10         thread.start();//一定要启动该线程
11         Handler handler = new Handler(thread.getLooper()){
12             @Override
13             public void handleMessage(Message msg) {
14                 Log.i(TAG,"handler thread:"+Thread.currentThread());
15                 switch (msg.what){
16                     case FLAG_TEST:
17                         //耗时操作...
18                         break;
19                     default:
20                         break;
21                 }
22                  super.handleMessage(msg);
23             }
24         };
25         handler.sendEmptyMessage(FLAG_TEST);
26     }

    log:

com.example.administrator.handlertest I/MainActivity﹕ main thread:Thread[main,5,main]
com.example.administrator.handlertest I/MainActivity﹕ handler thread:Thread[handler thread,5,main]

  通过log可以看出handler处在一个子线程中,这样就能够执行一些耗时操作。

  第十一行通过thread.getLooper()来创建handler,那么我们来看下getLooper()里面的源码:

 1 public Looper getLooper() {
 2         if (!isAlive()) {
 3             return null;
 4         }
 5
 6         // If the thread has been started, wait until the looper has been created.
 7         synchronized (this) {
 8             while (isAlive() && mLooper == null) {
 9                 try {
10                     wait();
11                 } catch (InterruptedException e) {
12                 }
13             }
14         }
15         return mLooper;
16     }

    看第8行代码,如果这个线程可用并且looper为null时,就会调用wait()方法,处于等待状态,这样可以有效的避免多线程并发操作引起的空指针异常。在thread启动时,会调用run()方法,再来看看run()方法里面的代码:

    

 1 @Override
 2     public void run() {
 3         mTid = Process.myTid();
 4         Looper.prepare();
 5         synchronized (this) {
 6             mLooper = Looper.myLooper();
 7             notifyAll();
 8         }
 9         Process.setThreadPriority(mPriority);
10         onLooperPrepared();
11         Looper.loop();
12         mTid = -1;
13     }

  第4行创建了Looper对象,第6、7行获取当前线程Looper之后调用notifyAll()方法。这时调用getLooper()方法返回一个Looper对象。 

 

  上面有提到使用HandlerThread避免多线程并发操作引起的空指针异常,这里解释下为什么:如果onCreate方法第11行通过程序员自定义的一个新线程创建handler时,很可能出现这样一个结果:创建handler的代码已经执行了,而新线程却还没有Looper.prepare()(创建Looper对象,那么这样就会导致空指针异常)。

  对代码稍做修改:

  

 1 package com.example.administrator.handlertest;
 2
 3 import android.os.Bundle;
 4 import android.os.Handler;
 5 import android.os.Looper;
 6 import android.os.Message;
 7 import android.support.v7.app.ActionBarActivity;
 8 import android.util.Log;
 9
10 public class MainActivity extends ActionBarActivity {
11
12     private static final String TAG = "MainActivity";
13     private static final int FLAG_TEST = 1;
14
15     @Override
16     protected void onCreate(Bundle savedInstanceState) {
17         super.onCreate(savedInstanceState);
18         setContentView(R.layout.activity_main);
19         Log.i(TAG,"main thread:"+Thread.currentThread());
20 //        HandlerThread thread = new HandlerThread("handler thread");
21 //        thread.start();//一定要启动该线程
22         MyThread thread = new MyThread();
23         thread.start();
24         Handler handler = new Handler(thread.looper){
25             @Override
26             public void handleMessage(Message msg) {
27                 Log.i(TAG,"handler thread:"+Thread.currentThread());
28                 switch (msg.what){
29                     case FLAG_TEST:
30                         //耗时操作...
31                         break;
32                     default:
33                         break;
34                 }
35                  super.handleMessage(msg);
36             }
37         };
38         handler.sendEmptyMessage(FLAG_TEST);
39     }
40
41     static class MyThread extends Thread{
42         Looper looper;
43         @Override
44         public void run() {
45             Looper.prepare();looper = Looper.myLooper();
46             //...
47             Looper.loop();
48         }
49     }
50 }

    运行结果:

1  Caused by: java.lang.NullPointerException
2             at android.os.Handler.<init>(Handler.java:234)
3             at android.os.Handler.<init>(Handler.java:142)
4             at com.example.administrator.handlertest.MainActivity$1.<init>(MainActivity.java:24)
5             at com.example.administrator.handlertest.MainActivity.onCreate(MainActivity.java:24)
6             at android.app.Activity.performCreate(Activity.java:5211)
7             at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1151)
8             at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2341)

    从异常信息第4行中可以看出:onCreate()方法第24行thread.looper是一个null.这时因为还没等新线程创建Looper,Handler就已经创建了。如果在第23行thread.start()后面休眠几秒就不会报空指针异常了。

最后补充一点,Android判断当前更新UI的线程是否是主线程的对象ViewRootImpl对象在onResume()中,所以只要子线程在onResume()之前完成更新UI也是能够实现的。这里只是简单提一下,知道就行,不过不要这么做。

    

    

时间: 2024-10-13 04:47:41

Android Handler消息传递机制详解的相关文章

Android Handler消息处理机制详解

前言 从我们学习android开始,几乎每天都在和handler打交道.有了它,我们在子线程中处理好了耗时的操作,可以利用它来更新UI.它为我们在线程间的通信提供了很大的方便,而今天博客就来详细的介绍一下Handler的消息循环机制,一步一步的了解其中的奥妙,本文不介绍Handler的详细使用,探究的是内部的原理.所以看这篇博客的童鞋需要有handler的基本使用能力 本博客基于Android 6.0源码讲解 先抛出一个简单的使用例子 public class DemoAct extends A

Android事件分发机制详解(1)----探究View的事件分发

探究View的事件分发 在Activity中,只有一个按钮,注册一个点击事件 [java] view plaincopy button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { Log.d("TAG", "onClick execute"); } }); 如果在需要一个触摸事件 [java] view plaincopy button.setO

Android事件分发机制详解(2)----分析ViewGruop的事件分发

首先,我们需要 知道什么是ViewGroup,它和普通的View有什么区别? ViewGroup就是一组View的集合,它包含很多子View和ViewGroup,是Android 所有布局的父类或间接父类. 但ViewGroup也是一个View,只不过比起View,它可以包含子View和定义布局参数的功能. 现在,通过一个Demo演示Android中ViewGroup的事件分发机制. 首先我们来自定义一个布局,命名为MyLayout,继承自LinearLayout,如下 所示: public c

Android事件传递机制详解及最新源码分析——ViewGroup篇

在上一篇<Android事件传递机制详解及最新源码分析--View篇>中,详细讲解了View事件的传递机制,没掌握或者掌握不扎实的小伙伴,强烈建议先阅读上一篇. 好了,废话还是少说,直奔主题,开始本篇的ViewGroup事件传递机制探索之旅. 依然从简单的Demo例子现象开始分析 新建安卓工程,首先自定义一个Button以及一个RelativeLayout,很简单,只是重写了主要与事件传递机制相关的方法,代码如下: 自定义WLButton类: 1 public class WLButton e

Android内存管理机制详解 (zhuan)

http://www.2cto.com/kf/201212/175786.html 与windows内存区别 在Linux中经常发现空闲内存很少,似乎所有的内存都被系统占用了,表面感觉是内存不够用了,其实不然.这是Linux内存管理的一个优秀特性,在这方面,区别于 Windows的内存管理.主要特点是,无论物理内存有多大,Linux都将其充份利用,将一些程序调用过的硬盘数据读入内存,利用内存读写的高速特性来提高Linux系统的数据访问性能.而Windows是只在需要内存时,才为应用程序分配内存,

Android 接口回调机制详解

在使用接口回调的时候发现了一个经常犯的错误,就是回调函数里面的实现有可能是用多线程或者是异步任务去做的,这就会导致我们期望函数回调完毕去返回一个主函数的结果,实际发现是行不通的,因为如果回调是多线程的话你是无法和主函数同步的,也就是返回的数据是错误的,这是非常隐秘的一个错误.那有什么好的方法去实现数据的线性传递呢?先介绍下回调机制原理. 回调函数 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数

Android Handler消息传递机制

1. Handler消息传递机制初步认识:什么是Handler? handler通俗一点讲就是用来在各个线程之间发送数据的处理对象.在任何线程中,只要获得了另一个线程的handler,则可以通过 handler.sendMessage(message)方法向那个线程发送数据.基于这个机制,我们在处理多线程的时候可以新建一个thread,这个thread拥有UI线程中的一个handler.当thread处理完一些耗时的操作后通过传递过来的handler向UI线程发送数据,由UI线程去更新界面. 主

Android 事件分发机制详解

更多内容请参照我的个人站点: http://stackvoid.com/ 网上很多关于Android事件分发机制的解释,大多数描述的都不够清晰,没有吧来龙去脉搞清楚,本文将带你从Touch事件产生到Touch事件被消费这一全过程作全面的剖析. 产生Touch事件 这部分牵扯到硬件和Linux内核部分:我们简单讲述一下这部分内容,如果有兴趣的话可以参考这篇文章. 传递Touch事件 触摸事件是由Linux内核的一个Input子系统来管理的(InputManager),Linux子系统会在/dev/

Android事件分发机制详解:史上最全面、最易懂

前言 Android事件分发机制是每个Android开发者必须了解的基础知识 网上有大量关于Android事件分发机制的文章,但存在一些问题:内容不全.思路不清晰.无源码分析.简单问题复杂化等等 今天,我将全面总结Android的事件分发机制,我能保证这是市面上的最全面.最清晰.最易懂的 本文秉着"结论先行.详细分析在后"的原则,即先让大家感性认识,再通过理性分析从而理解问题: 所以,请各位读者先记住结论,再往下继续看分析: 文章较长,阅读需要较长时间,建议收藏等充足时间再进行阅读 目