一.引言
经常在Android的java代码中动态设置布局的读者应该会对动态获取控件宽高不陌生,在最近项目中我也有用到。我们知道直接在onCreate
方法中无法通过getWidth
或getHeight
获取到想要的控件的宽高具体值。因此有几种方式来获取,例如在监听中获取或者通过View.post
的方式获取。具体方法可以参考下面的这篇文章,写得比较仔细。
Activity启动过程中获取组件宽高的五种方式
笔者也有使用View.post
来获取,不过始终对这个方法心存疑问,为什么一个类似handler.post的方法调用之后就可以正确得到宽高?通过查看源码以及查阅的一些资料大致弄懂了流程,接下来说一下我的分析,如果有不对的地方欢迎指教~
二.View.post方法的调用机制
View的post和postDelay方法其实是类似的,我们点进View.java中的这两个方法看一下:
public boolean post(Runnable action) {
final AttachInfo attachInfo = mAttachInfo;
if (attachInfo != null) {
return attachInfo.mHandler.post(action);
}
// Postpone the runnable until we know on which thread it needs to run.
// Assume that the runnable will be successfully placed after attach.
getRunQueue().post(action);
return true;
}
...
public boolean postDelayed(Runnable action, long delayMillis) {
final AttachInfo attachInfo = mAttachInfo;
if (attachInfo != null) {
return attachInfo.mHandler.postDelayed(action, delayMillis);
}
// Postpone the runnable until we know on which thread it needs to run.
// Assume that the runnable will be successfully placed after attach.
getRunQueue().postDelayed(action, delayMillis);
return true;
}
这两个方法中的代码基本是一致的,不同的只是postDelay中有将delayMillis传进去,而在post中最终也是调用了postDelay这个方法,只是将delayMillis置为0了,我们可以点进getRunQueue().post()
方法中看看:
可以看到确实如此。
接下来分析post中具体代码。代码比较简洁,我们可以看到其中有对
mAttachInfo
赋值的attachInfo
进行判断,如果为空,则调用getRunQueue().post()
,否则直接返回attachInfo.mHandler.post(action)
。那么这两个post有什么区别?
1.先来看当mAttachInfo
不为null时的情况,因为这个较为简单,点进attachInfo.mHandler.post
发现其实就是调用的Handler.post()
。这里就要看看这里的mHandler
是哪里产生的。使用AS的ctrl+左键一直追根溯源:
final static class AttachInfo {
...
final Handler mHandler;
...
AttachInfo(IWindowSession session, IWindow window, Display display,
ViewRootImpl viewRootImpl, Handler handler, Callbacks effectPlayer,
Context context) {
mSession = session;
mWindow = window;
mWindowToken = window.asBinder();
mDisplay = display;
mViewRootImpl = viewRootImpl;
mHandler = handler;
mRootCallbacks = effectPlayer;
mTreeObserver = new ViewTreeObserver(context);
}
可以看到mHandler是AttachInfo的一个变量,在AttchInfo的构造方法中被赋值,查看这个构造方法的调用点(这里可以结合Source Insight的ctrl+/的查找快捷键进行查找):
public final class ViewRootImpl implements ViewParent,
View.AttachInfo.Callbacks, ThreadedRenderer.DrawCallbacks {
...
mAttachInfo = new View.AttachInfo(mWindowSession, mWindow, display, this, mHandler, this,
context);
可以看到是在ViewRootImpl方法中将mHandler传入,查看mHanlder被赋值的地方:
final ViewRootHandler mHandler = new ViewRootHandler();
ViewRootImpl是在主线程中被创建,因此这个handler对象是主线程的handler,至此我们可以知道mAttachInfo
不为空的时候其实是直接调用了主线程handler来处理我们post的消息。当然这里有个问题是为什么使用了主线程的handler来处理消息之后就可以获取正确的宽高呢?先别急,我们继续梳理了之后来解释这个问题~
2.上面已经分析了mAttachInfo
不为空的情况,当mAttachInfo
为空时,会调用getRunQueue()
这个方法:
private HandlerActionQueue getRunQueue() {
if (mRunQueue == null) {
mRunQueue = new HandlerActionQueue();
}
return mRunQueue;
}
返回了一个HandlerActionQueue
的实例。这个HandlerActionQueue
是什么,点进去看看:
public class HandlerActionQueue {
private HandlerAction[] mActions;
private int mCount;
public void post(Runnable action) {
postDelayed(action, 0);
}
public void postDelayed(Runnable action, long delayMillis) {
final HandlerAction handlerAction = new HandlerAction(action, delayMillis);
synchronized (this) {
if (mActions == null) {
mActions = new HandlerAction[4];
}
mActions = GrowingArrayUtils.append(mActions, mCount, handlerAction);
mCount++;
}
}
...
private static class HandlerAction {
final Runnable action;
final long delay;
public HandlerAction(Runnable action, long delay) {
this.action = action;
this.delay = delay;
}
public boolean matches(Runnable otherAction) {
return otherAction == null && action == null
|| action != null && action.equals(otherAction);
}
}
}
这个类包含了两个全局变量,一个应该是用来计数,另一个mActions
则是一个HandlerAction
数组,HandlerAction
中封装了一个Runnable
对象和一个延时delay
。
继续看,当调用了post之后,实际只是生成了一个默认长度为4的HandlerAction
数组,将要实现的Runnable
传入。那我们什么时候使用Runnable
呢?我们可以注意到这个类中有个executeActions
方法:
public void executeActions(Handler handler) {
synchronized (this) {
final HandlerAction[] actions = mActions;
for (int i = 0, count = mCount; i < count; i++) {
final HandlerAction handlerAction = actions[i];
handler.postDelayed(handlerAction.action, handlerAction.delay);
}
mActions = null;
mCount = 0;
}
}
这个方法就是将我们post
进来的Runnable
使用handler进行处理,可以查看一下这个方法在哪里被调用以及handler是来自哪里。在View中可以查找到该方法在View.java
类中的dispatchAttachedToWindow
方法中被调用:
void dispatchAttachedToWindow(AttachInfo info, int visibility) {
...
if (mRunQueue != null) {
mRunQueue.executeActions(info.mHandler);
mRunQueue = null;
}
...
}
这里可以看到传入的就是AttachInfo 的mHandler变量,上面我们分析过最终mHandler就是主线程的Handler,那么现在的问题就是这个info是从哪里传入的。继续追查dispatchAttachedToWindow
方法使用的地方(Source Insight):可以看到总共有四个类有调用这个方法:AttachInfo_Accessor
,View
,ViewGroup
,ViewRootImpl
。
其中
AttachInfo_Accessor.java
类中没有我们需要关注的地方,View.java
就是当前类,显然没有被调用的地方,进入ViewGroup.java
看看:
@Override
void dispatchAttachedToWindow(AttachInfo info, int visibility) {
mGroupFlags |= FLAG_PREVENT_DISPATCH_ATTACHED_TO_WINDOW;
super.dispatchAttachedToWindow(info, visibility);
mGroupFlags &= ~FLAG_PREVENT_DISPATCH_ATTACHED_TO_WINDOW;
final int count = mChildrenCount;
final View[] children = mChildren;
for (int i = 0; i < count; i++) {
final View child = children[i];
child.dispatchAttachedToWindow(info,
combineVisibility(visibility, child.getVisibility()));
}
final int transientCount = mTransientIndices == null ? 0 : mTransientIndices.size();
for (int i = 0; i < transientCount; ++i) {
View view = mTransientViews.get(i);
view.dispatchAttachedToWindow(info,
combineVisibility(visibility, view.getVisibility()));
}
}
可以看到ViewGroup
的dispatchAttachedToWindow
方法中,调用了View
的dispatchAttachedToWindow
方法,将父中的AttachInfo
全部传入子类child中实现赋值,而此时的child就是View
类,我们不是要查看View
类中的这个方法在哪里调用吗?这里貌似陷入死循环了。。没事,我们在看最后的ViewRootImpl
方法中有没有什么信息:
private void performTraversals() {
...
host.dispatchAttachedToWindow(mAttachInfo, 0);
...
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
...
performLayout(lp, mWidth, mHeight);
...
performDraw();
...
}
在performTraversals
方法中我们看到了这个方法。host是由mView赋值,而mView就是由顶层视图DecorView所赋值的。这个performTraversals
方法就是用来调用测量、布局、绘制方法的地方。mAttachInfo唯一赋值的地方上面我们也有分析过:
mAttachInfo = new View.AttachInfo(mWindowSession, mWindow, display, this, mHandler, this,
context);
因此到这里我们就可以梳理一下了:在Activity的DecorView中调用dispatchAttachedToWindow
时,将mAttachInfo
传入到View
中,并调用mRunQueue.executeActions
的方法,该方法使用mAttachInfo
的handler
将我们最初调用View.post(runnable)
方法post进来的消息post到消息队列中进行处理,并且我们知道mAttachInfo
的handler
是主线程的handler
,因此其实就是post到了主线程的消息队列中等待处理。这里我们也就分析完毕。
不过这里有两个问题:
(1)前面我们知道由mAttachInfo
的值来决定调用哪个post。那么mAttachInfo什么时候不为空?查看mAttachInfo
被赋值的地方有两处:
void dispatchAttachedToWindow(AttachInfo info, int visibility) {
...
mAttachInfo = info;
...
}
void dispatchDetachedFromWindow() {
...
mAttachInfo = null;
...
}
从这里我们可以看到在dispatchAttachedToWindow
中mAttachInfo
被赋值为info
,这里的info
不就是从performTraversals
方法中调用的时候传入的吗~所以就是当attachedToWindow的时候被赋值,detachedFromWindow时被置为空。当mAttachInfo
为空的时候将Runnable
存放到HandlerAction
中,当View
的dispatchAttachedToWindow
方法被调用时使用主线程handler
将其post到消息队列中。当mAttachInfo
不为空的时候直接调用主线程handler
即可。
(2)在performTraversals
方法中我们是先调用dispatchAttachedToWindow
方法之后才开始调用measure
和layout
等方法进行测量布局的,而在dispatchAttachedToWindow
中我们就有调用了handler
将消息post到消息队列了准备执行了,那此时我们为什么能够在View.post
方法中获取到正确的长宽呢?
我们来查看performTraversals
的调用时机:
void doTraversal() {
...
performTraversals();
...
}
就一处被调用,继续看doTraversal
被调用的地方:
final class TraversalRunnable implements Runnable {
@Override
public void run() {
doTraversal();
}
}
我们可以看到其实它是在一个Runnable
中的run
方法中被调用,这个TraversalRunnable
方法被实例化之后,
final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
分别在scheduleTraversals
和unscheduleTraversals
方法中被调用,这两个方法直观上看上去就是成对出现的方法,我们来看scheduleTraversals
方法中有一行代码:
void scheduleTraversals() {
...
mChoreographer.postCallback(
Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
...
}
}
这个postCallback
方法貌似也是post
的近亲?我们查看Choreographer
中的这个方法可以看到其实最终就是调用了Message msg = mHandler.obtainMessage(MSG_DO_SCHEDULE_CALLBACK, action);
方法。这个mHandler
就是主线程Handler
~
因此谜底揭开,首先在主线程的Handler
中就已经post
进去了一个Runnable
来执行performTraversals
方法,当然就按照顺序执行了post我们调用View.post(runnable)
方法到主线程Handler
、测量、布局、绘制等一系列操作。然而由于Handler
的机制,它是将所有的message都post到一个MessageQueue
中,按照顺序执行这些消息。因此只有当执行完测量、布局、绘制之后,才能执行我们的Runnable
,所以我们这时就能够获取到正确的宽高了~
三.感想
第一次分析源码确实感觉很多知识点都不太理解,不过希望自己能坚持下来不断前行~分析的过程中有借鉴两位大神的文章,很感谢_
1.【Andorid源码解析】View.post() 到底干了啥
2.通过View.post()获取View的宽高引发的两个问题