0,参考:
Android_8.0.0 源码
分析 view.post() 为什么能够 获取 View的宽高
1,核心内容
/**
* main 在主线程调用
*/
public static void main() {
LogUtil.v("1");
new Handler(Looper.getMainLooper()).post(new Runnable() {
@Override
public void run() {
LogUtil.v("2");
}
});
LogUtil.v("3");
try {
Thread.sleep(1000); // 模拟一个超级耗时的操作
} catch (InterruptedException e) {
e.printStackTrace();
}
LogUtil.v("4");
}
问题:在主线程 调用 「main方法」,打印顺序是怎么样的,会不会受到延时的影响
结论:1 -> 3 -> 4 -> 2
原因:一个方法为一个执行单元。系统调用到「main方法」时,post 将 runnable 插入到Looper队列中。因此,runnable 必须等待「main方法」执行完成,才会被执行
2,获取宽高的原因【1基础上的实战】
ViewRootImpl.java中,performTraversals方法,该方法也是分发measure、layout、draw的实际入口
private void performTraversals() {
// 实际可能 post Runnable 的两个地方
host.dispatchAttachedToWindow(mAttachInfo, 0); // line:1658
getRunQueue().executeActions(mAttachInfo.mHandler); // line:1697
// 实际分发实现
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec); // line:2155
performLayout(lp, mWidth, mHeight); // line:2200
performDraw(); // line:2347
}
调用顺序如代码中所示,执行 “放入runnable” 的代码,是在 performMeasure、performLayout、performDraw 这三个方法之前的。而我们能在 Runnable 中,获取到测量完成之后的宽高,就是因为 Runnable 被 插入到 Looper 队列中。因此,runnable 被执行的代码,是在 performTraversals 方法结束之后,才被调用的。
因此,runnable 中的代码,就能够获取到 performTraversal 执行完成之后的数值。即:宽、高等数据