`
252190908
  • 浏览: 245527 次
文章分类
社区版块
存档分类
最新评论

Android源码分析-消息队列和Looper

 
阅读更多
转载请注明出处:http://blog.csdn.net/singwhatiwanna/article/details/17361775

前言

上周对Android中的事件派发机制进行了分析,这次博主要对消息队列和Looper的源码进行简单的分析。大家耐心看下去,其实消息队列的逻辑比事件派发机制简单多了,所以大家肯定会很容易看懂的。

概念

1. 什么是消息队列

消息队列在android中对应MessageQueue这个类,顾名思义,消息队列中存放了大量的消息(Message)

2.什么是消息

消息(Message)代表一个行为(what)或者一串动作(Runnable),有两处会用到Message:Handler和Messenger

3.什么是Handler和Messenger

Handler大家都知道,主要用来在线程中发消息通知ui线程更新ui。Messenger可以翻译为信使,可以实现进程间通信(IPC),Messenger采用一个单线程来处理所有的消息,而且进程间的通信都是通过发消息来完成的,感觉不能像AIDL那样直接调用对方的接口方法(具体有待考证),这是其和AIDL的主要区别,也就是说Messenger无法处理多线程,所有的调用都是在一个线程中串行执行的。Messenger的典型代码是这样的:new Messenger(service).send(msg),它的本质还是调用了Handler的sendMessage方法

4.什么是Looper

Looper是循环的意思,它负责从消息队列中循环的取出消息然后把消息交给目标处理

5.线程有没有Looper有什么区别?

线程如果没有Looper,就没有消息队列,就无法处理消息,线程内部就无法使用Handler。这就是为什么在子线程内部创建Handler会报错:"Can't create handler inside thread that has not called Looper.prepare()",具体原因下面再分析。

6.如何让线程有Looper从而正常使用Handler?

在线程的run方法中加入如下两句:

Looper.prepare();

Looper.loop();

这一切不用我们来做,有现成的,HandlerThread就是带有Looper的线程。

想用线程的Looper来创建Handler,很简单,Handler handler = new Handler(thread.getLooper()),有了上面这几步,你就可以在子线程中创建Handler了,好吧,其实android早就为我们想到这一点了,也不用自己写,IntentService把我们该做的都做了,我们只要用就好了,具体怎么用后面再说。

消息队列和Looper的工作机制

一个Handler会有一个Looper,一个Looper会有一个消息队列,Looper的作用就是循环的遍历消息队列,如果有新消息,就把新消息交给它的目标处理。每当我们用Handler来发送消息,消息就会被放入消息队列中,然后Looper就会取出消息发送给它的目标target。一般情况,一个消息的target是发送这个消息的Handler,这么一来,Looper就会把消息交给Handler处理,这个时候Handler的dispatchMessage方法就会被调用,一般情况最终会调用Handler的handleMessage来处理消息,用handleMessage来处理消息是我们常用的方式。

源码分析

1. Handler发送消息的过程

    public final boolean sendMessage(Message msg)
    {
        return sendMessageDelayed(msg, 0);
    }
	
	public final boolean sendMessageDelayed(Message msg, long delayMillis)
    {
        if (delayMillis < 0) {
            delayMillis = 0;
        }
        return sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis);
    }
	
	public boolean sendMessageAtTime(Message msg, long uptimeMillis) {
		MessageQueue queue = mQueue;
		if (queue == null) {
			RuntimeException e = new RuntimeException(
					this + " sendMessageAtTime() called with no mQueue");
			Log.w("Looper", e.getMessage(), e);
			return false;
		}
		return enqueueMessage(queue, msg, uptimeMillis);
	}
	
	private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {
        msg.target = this;
        if (mAsynchronous) {
            msg.setAsynchronous(true);
        }
        //这里msg被加入消息队列queue
        return queue.enqueueMessage(msg, uptimeMillis);
    }

2.Looper的工作过程

   public static void loop() {
        final Looper me = myLooper();
        if (me == null) {
            throw new RuntimeException("No Looper; Looper.prepare() wasn't called on this thread.");
        }
        //从Looper中取出消息队列
        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 (;;) {
            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);
            }

            //将消息交给target处理,这个target就是Handler类型
            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();
        }
    }

3.Handler如何处理消息

    /**
     * Subclasses must implement this to receive messages.
     */
    public void handleMessage(Message msg) {
    }
    
    /**
     * Handle system messages here.
     */
    public void dispatchMessage(Message msg) {
        if (msg.callback != null) {
            //这个方法很简单,直接调用msg.callback.run();
            handleCallback(msg);
        } else {
            //如果我们设置了callback会由callback来处理消息
            if (mCallback != null) {
                if (mCallback.handleMessage(msg)) {
                    return;
                }
            }
            //否则消息就由这里来处理,这是我们最常用的处理方式
            handleMessage(msg);
        }
    }
我们再看看msg.callback和mCallback是啥东西

/*package*/ Runnable callback;

现在已经很明确了,msg.callback是个Runnable,什么时候会设置这个callback:handler.post(runnable),相信大家都常用这个方法吧

 /**
 * Callback interface you can use when instantiating a Handler to avoid
 * having to implement your own subclass of Handler.
 *
 * @param msg A {@link android.os.Message Message} object
 * @return True if no further handling is desired
 */
public interface Callback {
	public boolean handleMessage(Message msg);
}
	
final Callback mCallback;

而mCallback是个接口,可以这样来设置 Handler handler = new Handler(callback),这个callback的意义是什么呢,代码里面的注释已经说了,可以让你不用创建Handler的子类但是还能照样处理消息,恐怕大家常用的方式都是新new一个Handler然后override其handleMessage方法来处理消息吧,从现在开始,我们知道,不创建Handler的子类也可以处理消息。多说一句,为什么创建Handler的子类不好?这是因为,类也是占空间的,一个应用class太多,其占用空间会变大,也就是应用会更耗内存。

HandlerThread简介

    @Override
    public void run() {
        mTid = Process.myTid();
        Looper.prepare();
        synchronized (this) {
            mLooper = Looper.myLooper();
            notifyAll();
        }
        Process.setThreadPriority(mPriority);
        onLooperPrepared();
        Looper.loop();
        mTid = -1;
    }
HandlerThread继承自Thread,其在run方法内部为自己创建了一个Looper,使用上HandlerThread和普通的Thread不一样,无法执行常见的后台操作,只能用来处理新消息,这是因为Looper.loop()是死循环,你的code根本执行不了,不过貌似你可以把你的code放在super.run()之前执行,但是这好像不是主流玩法,所以不建议这么做。

IntentService简介

    public void onCreate() {
        // TODO: It would be nice to have an option to hold a partial wakelock
        // during processing, and to have a static startService(Context, Intent)
        // method that would launch the service & hand off a wakelock.

        super.onCreate();
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        thread.start();

        mServiceLooper = thread.getLooper();
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }
IntentService继承自Service,它是一个抽象类,其被创建的时候就new了一个HandlerThread和ServiceHandler,有了它,就可以利用IntentService做一些优先级较高的task,IntentService不会被系统轻易杀掉。使用IntentService也是很简单,首先startService(intent),然后IntentService会把你的intent封装成Message然后通过ServiceHandler进行发送,接着ServiceHandler会调用onHandleIntent(Intent intent)来处理这个Message,onHandleIntent(Intent intent)中的intent就是你startService(intent)中的intent,ok,现在你需要做的是从IntentService派生一个子类并重写onHandleIntent方法,然后你只要针对不同的intent做不同的事情即可,事情完成后IntentService会自动停止。所以,IntentService是除了Thread和AsyncTask外又一执行耗时操作的方式,而且其不容易被系统干掉,建议关键操作采用IntentService。

在子线程创建Handler为什么会报错?

    public Handler(Callback callback, boolean async) {
        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();
        //报错的根本原因是:当前线程没有Looper
        if (mLooper == null) {
            throw new RuntimeException(
                "Can't create handler inside thread that has not called Looper.prepare()");
        }
        mQueue = mLooper.mQueue;
        mCallback = callback;
        mAsynchronous = async;
    }
如何避免这种错误:在ui线程使用Handler或者给子线程加上Looper。
分享到:
评论

相关推荐

    Android异步消息处理机制详解及源码分析 - 工匠若水 - 博客频道 - CSDN1

    【Android异步消息处理机制详解及源码分析】 在Android应用开发中,为了保证用户界面的流畅性,我们需要避免在主线程(UI线程)执行耗时操作。Android提供了异步消息处理机制,由Handler、Message、MessageQueue和...

    android的消息处理机制(图+源码分析)——Looper,Handler,Message

    android源码中包含了大量的设计模式, 除此以外, android sdk还精心为我们设计了各种helper类, 对于和我一样渴望水平得到进阶的人来说, 都太值得一读了。 这不, 前几天为了了解android的消息处理机制, 我看 了...

    消息循环 Looper 及其源码解析

    在Android系统中,消息处理机制是应用程序中线程间通信的一种关键方式,它涉及到Handler、Message和Looper等组件。本文将深入探讨“消息循环”Looper及其源码解析,帮助开发者更好地理解和应用这一机制。 首先,...

    安卓Android源码——HandlerLooper2.rar

    这个压缩包“安卓Android源码——HandlerLooper2.rar”可能包含了关于这些组件的深入分析和示例代码。以下是关于`Handler`、`Looper`和`MessageQueue`的详细解释: 1. **Handler**: - `Handler`是Android中的一个...

    Android线程模式(handler,thread,looper)

    本文将深入探讨Android中的三种主要线程模式:Handler、Thread以及Looper,并结合源码分析它们的工作原理。 首先,我们来理解一下Android应用的基本运行环境。Android系统默认在主线程(UI线程)中执行所有的用户...

    安卓Android源码——HandlerLooper1.rar

    2. `Looper`的初始化和消息循环的启动过程。 3. `MessageQueue`的工作原理,以及消息的添加、删除和调度。 4. 实战案例,如使用`Handler`实现定时任务、延迟任务和取消任务。 5. `Handler`与`AsyncTask`、`Runnable`...

    Android应用源码之HandlerLooper1_Android.zip

    在解压的文件“Android应用源码之HandlerLooper1”中,可能包含了一个简单的Android工程,演示了如何创建`Handler`,使用`Looper`启动消息循环,以及发送和处理消息。通过阅读和分析这个源码,你可以更直观地了解这...

    Android应用源码之HandlerLooper1.zip

    本资料“Android应用源码之HandlerLooper1.zip”应该是包含了一个关于这些组件的详细示例或分析,让我们来深入探讨它们的工作原理。 首先,`Handler`是Android中的一个类,它用于在UI线程中发送和处理消息。当你...

    进阶android源码demo

    "进阶Android源码demo"提供了一系列的实践项目,包括游戏和聊天应用,为开发者提供了丰富的学习资源。下面,我们将深入探讨这些知识点,以帮助你更好地理解和应用Android源码。 1. **Android框架理解**:Android...

    android2.2.3里面的handler源码,可用于学习,或用于java、javafx等项目,作为线程消息队列使用。.zip

    Android操作系统作为目前最广泛使用的移动设备操作系统之一,其内部机制和架构设计...通过分析和学习Android Handler源码,开发者可以更加深入地掌握线程消息队列的工作原理,从而设计出更加高效和稳定的多线程应用。

    Android即时通讯--仿QQ即时聊天源码(含服务器).zip

    8. **群聊功能**:群聊涉及到更复杂的用户管理和消息广播,开发者可以从源码中了解到如何实现群成员管理、群消息的分发等。 9. **界面设计与交互**:模仿QQ的聊天界面,可以学习到Android UI组件的使用,如...

    Android应用源码之HandlerLooper2.zip

    综上所述,`Android应用源码之HandlerLooper2.zip`中的示例可能展示了如何有效地使用`Handler`、`Looper`和`MessageQueue`来管理线程通信和异步任务。通过分析和学习这个示例代码,开发者可以更深入地理解Android...

    Handler源码分析流程以及面试问题解答

    Handler是Android系统中用于线程间通信的重要...以上是对Handler源码分析及面试问题解答的概述,深入理解这些知识点对于Android开发人员来说至关重要,有助于解决实际开发中遇到的问题,也能提高系统优化和调试的能力。

    Android应用源码之MessageManager-IT计算机-毕业设计.zip

    5. **Message对象**:包含了要传递的数据和消息代码,用于标识不同类型的消息。 6. **AsyncTask**或`Runnable`:在Android中,通常使用`AsyncTask`或者`Runnable`进行后台任务处理,它们可以与`Handler`配合,异步...

    androidjava源码-HandlerDemo:“深入了解Android消息机制和源码分析(Java层和Native层)”文章示例代码

    在Android源码中,可以看到`Looper_jni`相关的文件,这些文件定义了JNI方法,使得Java层的Looper可以与Native层的消息循环进行通信。\n\n在“HandlerDemo”项目中,你可以找到一系列的示例,演示了如何创建Handler、...

    Android 异步消息处理机制 让你深入理解 Looper、Handler、Message三者关系 - Hongyang -

    这篇文章将通过源码分析来详细解释这三个类之间的关系以及它们在Android中的作用。 1. **Looper** Looper是一个消息循环器,它维护了一个MessageQueue(消息队列),并持续不断地从队列中取出消息进行处理。在...

    关于Looper的使用

    1. **创建Message**:在需要发送消息时,通过Handler的obtainMessage()方法创建一个Message对象,设置消息的目标Handler和消息内容等。 2. **将Message放入队列**:调用Handler的sendMessage()或sendEmptyMessage()...

    Handler、Looper

    源码分析方面,Handler的`sendMessage()`方法最终会将Message插入到Looper的消息队列。这个队列是一个基于数组的双端队列(ArrayDeque)。Looper的`loop()`方法会不断调用`pollNext()`或`peekNext()`来获取Message。...

    【优亿Android特刊】第二十二期:Android源码修改

    Android源码分析总结** - **应用程序框架**:深入了解如何构建应用程序框架,例如Activity Manager、Package Manager等。 - **消息处理机制**:研究消息队列、Handler和Looper的工作原理。 - **系统原理**:探索...

    android handler 机制 源码

    源码分析: Handler的`sendMessage(Message msg)`方法最终会调用`enqueueMessage(Looper looper, Message msg, long when)`,将消息插入到Looper的消息队列中。而Looper的`loop()`方法则不断地从消息队列中取出...

Global site tag (gtag.js) - Google Analytics