自定义 view - 3大核心方法补充

自定义 view 的3个核心方法

  • onMeasure
    根据 view 的测量模式计算确定 view 的宽高
  • onLayout
    ViewGroup 中对所有的子 view 排版,决定子 view 的位置
  • onDraw
    具体绘制 view

要继续了解的点

我们在了解了自定义 view 的3大核心方法后就完了吗,没有啊,这只是开始呢,3大核心方法中还有一些要补充的点,这些我们清楚之后,会在今后的开发中帮助我们理顺思路:

  • view 的宽高生命周期内的变化
  • onSizeChange 方法是不是一定会执行
  • view 的位移,位置变化会触发 view 自身的那些函数
  • ViewGroup 会执行2次 onMeasure ,2次 onLayout ,1次 onDraw
  • 自定义的ViewGroup 的 setWillNotDraw(false)

ViewGroup 的绘制

在自定义 ViewGroup 中,我们需要 setWillNotDraw(false),才能执行 ViewGroup 自身的绘制方法

ViewGroup 的绘制过程如下:

public void draw(Canvas canvas) {
  . . . 
  // 绘制背景,只有dirtyOpaque为false时才进行绘制,下同
  int saveCount;
  if (!dirtyOpaque) {
    drawBackground(canvas);
  }

  . . . 

  // 绘制自身内容
  if (!dirtyOpaque) onDraw(canvas);

  // 绘制子View
  dispatchDraw(canvas);

   . . .
  // 绘制滚动条等
  onDrawForeground(canvas);

}

看见没有 ViewGroup 先绘制的背景,再绘制自身,最后遍历绘制子 view


view 的宽高生命周期内的变化

view 自身大小的失计算问题其实涉及 view 的 onMeasure 和 onSizeChange 方法,我们要在 view 的真个生命周期范围内观察 view 宽高的变化

重新回顾一下,view 的宽高涉及到2套 API:

  • getWidth() / getHeight():获得View最终的宽 / 高
  • getMeasuredWidth() / getMeasuredHeight():获得 View测量的宽 / 高

然后我们设计一个自定义 view 然后打印一下在 view 的各个生命周期方法中 宽高的数值

自定义 view

public class MyView2 extends View {

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);

        Log.d("AAA", "onMeasure /" + " widthSize = " + widthSize + " , heightSize = " + heightSize);
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);

        Log.d("AAA", "onSizeChanged / w = " + w + " , h = " + h + " , oldw = " + oldw + " , oldh = " + oldh);
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        super.onLayout(changed, left, top, right, bottom);

        Log.d("AAA", "onLayout / measureWidth = " + getMeasuredWidth() + " , measureHeight = " + getMeasuredHeight() + " , width = " + getWidth() + " , height = " + getHeight());
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        Log.d("AAA", "onLayout / measureWidth = " + getMeasuredWidth() + " , measureHeight = " + getMeasuredHeight() + " , width = " + getWidth() + " , height = " + getHeight());

    }
}

然后使用一个最简单的布局方案来观察

<FrameLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.bloodcrown.aaa02.MeasureActivity">

        <com.bloodcrown.aaa02.MyView2
            android:layout_width="match_parent"
            android:layout_height="match_parent"/>

</FrameLayout>

结果:


Snip20180627_4.png

从结果来看:

  • onMeasure 方法执行多次,其中计算出来的宽高值都可能会改变
  • onMeasure 方法中 getWidth、getHeight 方法的返回值都是0
  • view 宽高的改变触发了 onSizeChange 方法,onSizeChange 方法在 onLayout 之前执行
  • onLayout 方法中2套 API 的取值都是一样的,在这个时候 getWidth、getHeight 已经可以获取到具体的返回值

然后有一个问题,这里 onSizeChange 方法的执行是因为 onMeasure 执行了2次,view 的大小发生了改变。那么若是view 的大小不改变,那么 onSizeChange 还有机会执行吗

上面的例子中,自定义 view 使用了 match_parent 计算方法引起了 view 大小的变化,那么我们给一个具体值呢。

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.bloodcrown.aaa02.MeasureActivity">

        <com.bloodcrown.aaa02.MyView2
            android:layout_width="200dp"
            android:layout_height="200dp"/>

</FrameLayout>

结果:


Snip20180627_5.png

view 的大小没有变化,onSizeChange 方法 一样执行了,其实很好理解,view 的会记录自己的大小,在 view 初始化时 view 的大小是 0啊,所以 onSizeChange 方法是一定会执行的,至少会执行一次


view 的坐标属性变化和位移会触发 view 的哪些生命周期方法

1. UI 重绘方法

UI 重绘方法有下面几个方法:

  • invalidate()
    重绘视图,必须 Ui 线程执行
  • postInvalidate()
    重绘视图,使用 handle 消息机制允许异步执行
  • requestLayout()
    重新布局
  • requestFocus
    局部刷新,只重绘焦点部分或是我们指定的部分

然后我们来看下这几个方法会触发哪几个 view 的函数

  • invalidate


    Snip20180628_3.png

    这么看的话重绘真的是重绘,只会触发 view 的绘制方法

  • postInvalidate


    Snip20180628_5.png

    一样只会触发 view 的绘制方法,看来 postInvalidate 和 invalidate 区别的地方只是支不支持异步执行

  • requestLayout


    Snip20180628_2.png

    计算,布局,绘制 3个方法方法都触发了,看来是把 view 彻底重新走了一遍,大家想想也是,view 的位置和大小要是变化了呢。有的文章说 requestLayout 不会触发 onMeasure 方法,但是自己跑一下,还不是发现3个方法都执行了,也许是 android 版本的问题,这里我使用的是 API 26

  • requestFocus
    这个方法我试了下,不管是游参数的,没参数的都不会触发任何函数,这个方法我也不熟,姑且简单的直接执行了下,没触发任何 view 的函数我也是有些奇怪。就这样吧,哪位看到这里有了解的,请在评论里指点一下

2. view 位移

我们就来简单的设置下 translationY 来看看

结果是没有触发任何 view 的方法,简单翻翻 setTranslationY 方法的源码,里面调用的也是 invalidate 方法,但是没看到 view 的3个核心方法被触发,这么说 view 的位移并不是我们简单想想的 重新布局,具体啥怎么执行的,我也不知道啊,大家自己去研究把

3. view 位置变化

上面我们操作的是 view 的以为属性参数来移动的 view ,那么我们来试试 setTop 这类直接改变 view 坐标参数的方法


Snip20180628_7.png

view 的大小本质是由 left,top,buttom,right 4个参数决定的,所以我们改了这4个参数的任意一个就是改变了 view 的大小,可以看到 触发了 view 的 onSizeChange 大小改变函数,然后重新绘制,布局方法没有触发我有点想不通,大小改变了其实也算是 view 的位置改变了啊。


视图层级和多次测量,布局的关系

view 的计算,布局,绘制函数不是自己调用的,而是父控件 ViewGroup 决定何时,何地调用子 view 的相关方法

这里涉及到一个结论:

ViewGroup 绘制一次会调用子 view 的2次 onMeasure ,2次 onLayout ,1次 onDraw,部分 ViewGroup 具体子类会对自身测量2次,那么就会 调用子 view 的4次 onMeasure

官方文档有句话:

ViewGroup 会执行2 测量,第一次使用 USPENCIFIED 模式测量子 view 的真实大小,第二次使用 EXACTLY 模式再次测量子view

所以就产生了 2次 onMeasure ,2次 onLayout ,1次 onDraw。我打印了一下 view 2次测量,发现其实2次测量传入的测量模式参数都是一样的,具体原因有人说是 viewRootImpl 的原因,viewRootImpl 会执行2测测量,详细的自己去 Google 下

然后我测试了下,发现不同的 ViewGroup 子类中,根据子 view 的测量模式不同,会对子 view 的测量方法执行次数造成影响

FrameLayout 帧布局

view 不论用的哪种测量模式都会造成子 view 2次测量

ConstraintLayout 约束布局

view 使用 match_parent 会执行4测测量,使用具体的宽高值时,比如200dp,只会执行2次测量

match_parent


Snip20180629_9.png

200dp * 300dp


Snip20180629_10.png

从上面的打印信息中,可以清晰的看到 ConstraintLayout 约束布局中 match_parent 测量模式会测量出前后不同的高度参数。

RelativeLayout 相对布局

不管是 match_parent 还是 200dp ,都会触发自子 view 4次测量

match_parent


Snip20180629_16.png

200dp * 400dp


Snip20180629_14.png
LinearLayout 线性布局

同 FrameLayout 帧布局,只会2次测量

关于执行4次 view 的测量可以这样解释

某些 ViewGroup 子类会对 ViewGroup 自身执行2次测量,进而引起子 view 4次测量

在上面的打印日志中,可以看到 viewRootImpl 的身影,viewRootImpl 在第二次测量之后触发了一个大小改变的函数,然后接着是2次测量。可以猜测下是不同的 ViewGroup 子类有时候会触发 window 视图的变化,从而连带着造成 ViewGroup 的再次测量,ViewGroup 一次测量会对子 view 测量2次,那么 window 2次测量 ViewGroup 就会对子 view 测量4次


在多视图层级下,最内层的 view 执行的测量次数和父控件层数和类型的关系

那么继续深入,在多视图层级下,最内层的 view 会执行几次测量呢。

来个3层视图层级的例子,父视图都是 ConstraintLayout,view 是 match_parent 测量模式

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.bloodcrown.aaa02.MeasureActivity">

    <android.support.constraint.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <android.support.constraint.ConstraintLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">

            <com.bloodcrown.aaa02.MyView2
                android:id="@+id/view_my"
                android:layout_width="match_parent"
                android:layout_height="match_parent"/>

        </android.support.constraint.ConstraintLayout>
    </android.support.constraint.ConstraintLayout>
</android.support.constraint.ConstraintLayout>

结果:


Snip20180629_17.png

view 执行了16 次测量,16次 = 内层 4次 * 第二层视图 2次 * 第一层视图 2次,我这里用的都是 match_parent,外层 ViewGroup 都会对自己执行2遍测量,比如视图第二层的 ViewGroup ,对自己测量一遍就是最内层的 ViewGroup 对 view 测量4遍,那么 视图第二层的 ViewGroup 对自己测量2遍,就是 最内层的 ViewGroup 对 view 测量8遍,以此类推,就是 16遍

我用 systrace 工具专门看了下这一帧的执行情况


Snip20180629_22.png

systrace 工具自动分析已经开始提示昂贵的测量,布局损耗了,仅仅只是 16 次测量而已,就花了 6 毫秒,大家看到没 view 的 测量很耗费 cpu 资源的,所以我们应该尽可能的减少 view 的测量,优化布局性能任重而道远

我们把第二层视图 ViewGroup 改成 200dp * 200dp ,猜测下 view 应该会执行 4次 * 1次 * 2次,一共 8次测量

看结果:的确是 8次


Snip20180629_20.png

那我们把多有的上层 ViewGroup 都改成 200dp * 200dp

看结果:


Snip20180629_18.png

view 执行2次,等于 ViewGroup 只执行了一次测量,当然在实际中 ViewGroup 不可能都能有个具体的宽高值,但是在我们能够明确知道布局的宽高时,写具体数据是能够大大减少布局中 view 的测量次数的

这个话题就写到这里了,还有其他想法的同学自己试着玩吧。另外我们有兴趣的同学可以用 systrace 全程观察下


最后我们来看下 view 整个 acitivty 页面中的方法调用

20161030164616308.png

在 acitivty resume 之后 布局view 才会加入到 window 中,然后开始整个布局的测量,布局,绘制

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

推荐阅读更多精彩内容