关于View.post()/postDelay()方法的一些分析

一.引言

经常在Android的java代码中动态设置布局的读者应该会对动态获取控件宽高不陌生,在最近项目中我也有用到。我们知道直接在onCreate方法中无法通过getWidthgetHeight获取到想要的控件的宽高具体值。因此有几种方式来获取,例如在监听中获取或者通过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()方法中看看:

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_AccessorViewViewGroupViewRootImpl

image.png

其中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()));
        }
    }

可以看到ViewGroupdispatchAttachedToWindow方法中,调用了ViewdispatchAttachedToWindow方法,将父中的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的方法,该方法使用mAttachInfohandler将我们最初调用View.post(runnable)方法post进来的消息post到消息队列中进行处理,并且我们知道mAttachInfohandler是主线程的handler,因此其实就是post到了主线程的消息队列中等待处理。这里我们也就分析完毕。
不过这里有两个问题:
(1)前面我们知道由mAttachInfo的值来决定调用哪个post。那么mAttachInfo什么时候不为空?查看mAttachInfo被赋值的地方有两处:

void dispatchAttachedToWindow(AttachInfo info, int visibility) {
...
    mAttachInfo = info;
...
}
void dispatchDetachedFromWindow() {
...
        mAttachInfo = null;
...
}

从这里我们可以看到在dispatchAttachedToWindowmAttachInfo被赋值为info,这里的info不就是从performTraversals方法中调用的时候传入的吗~所以就是当attachedToWindow的时候被赋值,detachedFromWindow时被置为空。当mAttachInfo为空的时候将Runnable存放到HandlerAction中,当ViewdispatchAttachedToWindow方法被调用时使用主线程handler将其post到消息队列中。当mAttachInfo不为空的时候直接调用主线程handler即可。
(2)在performTraversals方法中我们是先调用dispatchAttachedToWindow方法之后才开始调用measurelayout等方法进行测量布局的,而在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();

分别在scheduleTraversalsunscheduleTraversals方法中被调用,这两个方法直观上看上去就是成对出现的方法,我们来看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的宽高引发的两个问题

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容