Handler 实际应用 - 获取View的宽高

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 执行完成之后的数值。即:宽、高等数据

 

 

 

 

 

 

你可能感兴趣的:(Android_源码,Android,Handler,View测量宽高)