转载请注明出处,Lee:http://blog.csdn.net/hnulwt/article/details/44457905
Handler有俩个主要作用:
1,to schedule messages and runnables to be executed as some point in the future。在未来的某个时候去调度messages或者要执行的runnables。
2, to enqueue an action to be performed on a different thread than your own。将一个要在另一个线程执行的动作加入队列。
下面这些方法完成了调度messages。
post
postAtTime(Runnable, long)
postDelayed
sendEmptyMessage
sendMessage
sendMessageAtTime
sendMessageDelayed
post相关方法
post方法允许你把一个Runnable对象加入到队列中,我们来看看他的源码实现:
/**
* @return Returns true if the Runnable was successfully placed in to the
* message queue. Returns false on failure, usually because the
* looper processing the message queue is exiting.
*/
public final boolean post(Runnable r)
{
return sendMessageDelayed(getPostMessage(r), 0);
}
这里返回true的话就说明这个Runnable对象被成功的放到Message Queue中,如果失败就返回的是false。
sendMessageDelayed发送的是一个Message对象,这里就可以看出他也是将Runnable对象转为Message对象的。来看看getPostMessage的实现:
private static Message getPostMessage(Runnable r) {
Message m = Message.obtain();
m.callback = r;
return m;
}
这里他从Message池中获取了一个Message对象m,将Runnable对象的地址给了m 包访问权限的Runnable对象,也就是说这里的Runnable其实是Message的一个成员变量。
getPostMessage方法执行完毕后,执行sendMessageDelayed方法,这个方法里面做了数值范围的检测,接着调用sendMessageAtTime:
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);
}
这里将mQueue传入方法enqueueMessage,这个mQueue是在Handler的构造函数中获取的,两句关键代码是,这里主要研究handler相关的,这个mQueue怎么创建的暂时不做深入:
mLooper = Looper.myLooper();
mQueue = mLooper.mQueue;
这里enqueueMessage最终调用MessageQueue类里面的enqueueMessage方法,最终传递msg和uptimeMillis给MessageQueue。
private boolean enqueueMessage(MessageQueue queue, Message msg, long uptimeMillis) {
msg.target = this;
if (mAsynchronous) {
msg.setAsynchronous(true);
}
return queue.enqueueMessage(msg, uptimeMillis);
}
分析完post方法,也就知道postAtTime、postDelayed等方法和他的实现类似。
removeCallBacks相关方法
相对于post方法,removeCallBacks方法调用层数就没那么多,他通过直接调用MessageQueue类里面的removeMessage达到目的。
/**
* Remove any pending posts of Runnable r that are in the message queue.
*/
public final void removeCallbacks(Runnable r)
{
mQueue.removeMessages(this, r, null);
}