自定义控件知识储备-View的绘制流程

在自定义控件这个学习系列里,首先写篇文章记录一下View的绘制流程,压压惊:-P。也为以后的自定义控件实践打个基础。虽然讲解View工作流程的文章很多,其中不乏很多精品文章,不过自己能从中理清思路,以自己之言总结出来,也是十分必要的。好的,我要开始装…不,总结了。

1. 前言

当我们打开手机,开始看朋友圈,刷微博的时候,我们有考虑过在我们眼前的一个个View是如何从无到有的展示在我们眼前的么?有考虑过它们的感受么?(神经病才去考虑(ノಠಠ)ノ彡┻━┻……)。

当我们在一张纸上画画的时候,哪怕是简单的一只小鸡,我们也不得不考虑下面几点:

  • 这只鸡得画多大呀?多宽,多高?不能大的超过纸的范围吧?
  • 这只鸡画在纸的哪里呢?是纸的中间还是靠下面一点呢?
  • 确定好大小和位置了,该怎么画呢?公鸡母鸡?这只鸡是什么形状(当然是鸡形)?什么颜色?

其实在屏幕上“画”一个View跟上述的流程也很相似。同样是经过了测量流程、布局流程以及绘制流程。我们都知道,Android界面布局是以一棵树的结构形式展现的,看我们的xml布局文件也看的出来。而绘制出整个界面肯定是要遍历整个View树,对这棵树的所有节点分别进行测量,布局和绘制。万事皆有源头,绘制这棵树得从根节点顶级View开始画起,也就是DecorView。至于啥是DecorView,大家可以自行去查阅资料。

系统内部会依次调用DecorView的measurelayoutdraw三大流程方法。measure方法又会调用onMeasure方法对它所有的子元素进行测量,如此反复调用下去就能完成整个View树的遍历测量。同样的,layoutdraw两个方法里也会调用相似的方法去对整个View树进行遍历布局和绘制。

下面就以这三个流程来了解一下View从无到有的不容易。

后退,我要开始装逼了

2. 测量流程-measure

测量流程得分情况来看,如果是单身View,那自然是没话说,自己照顾好自己,本分的测量好自己就行。而如果是为人父母的ViewGroup,那就得顾家了,除了测量好自己,还得去调用孩子们的measure方法让孩子们都测量好自己。甚至很多时候,ViewGroup得先测量好孩子们,最后才能确定自己的测量大小。一把辛酸泪…(ノへ ̄、)

下面分别来看看View和ViewGroup的测量过程:

2.1 View的measure过程

View类的measure方法的签名如下:

public final void measure(int widthMeasureSpec, int heightMeasureSpec)

看到这个方法,我得提出两个问题:

  1. 形参widthMeasureSpecheightMeasureSpec是几个意思?是用来测量自身大小的宽高么?
  2. measure方法是final修饰的,那怎么通过重写此方法来实现自定义控件的测量方式呢?

要回答第1个问题,首先得弄清楚:在界面的绘制过程中,View的这个方法是被它的父控件调用的,也就是说widthMeasureSpecheightMeasureSpec是通过父控件传递进来的,如果这两个参数是完全用来决定孩子View的大小,那孩子们也太没主动权了。

呵呵哒

事实上,这两个参数在很大程度上是决定了一个View的尺寸的,只不过孩子View可能各有各的特点,它们是能根据自身的特点来进行调整的,具体的呢以后再说。先来具体的看看MeasureSpec:

测量规格MeasureSpec

widthMeasureSpec这样的32位的int类型的数肯定是有自己的故事滴,它的高2位代表测量模式Mode,低30位代表测量大小Size。系统提供了一个MeasureSpec类来对这个参数进行操作,代码如下:

  public static class MeasureSpec {
   

        private static final int MODE_SHIFT = 30;
        private static final int MODE_MASK  = 0x3 << MODE_SHIFT;
        public static final int UNSPECIFIED = 0 << MODE_SHIFT;
        public static final int EXACTLY     = 1 << MODE_SHIFT;
        public static final int AT_MOST     = 2 << MODE_SHIFT;


        public static int makeMeasureSpec(int size, int mode) {
            if (sUseBrokenMakeMeasureSpec) {
                return size + mode;
            } else {
                return (size & ~MODE_MASK) | (mode & MODE_MASK);
            }
        }


        public static int getMode(int measureSpec) {
            return (measureSpec & MODE_MASK);
        }


        public static int getSize(int measureSpec) {
            return (measureSpec & ~MODE_MASK);
        }
    }

上面的代码也不复杂,都是通过位运算来进行操作的。(我在平时位运算用的少,所以我还得慢慢捋一捋才看的明白。╥﹏╥…)不过,这样做的好处就是更省内存,因为要是我来做的话,肯定是为这样的测量规格定义一个类,里面有mode和size两个属性,这样每次就会new很多测量规格的对象了。

好了,喝口水,接着往下说。既然测量规格是由测量模式mode和测量大小size组成的,size好说,那测量模式mode代表什么含义呢。由上面的代码可知,测量模式有三类:

  • UNSPECIFIED

    父控件不对你有任何限制,你想要多大给你多大,想上天就上天。这种情况一般用于系统内部,表示一种测量状态。(这个模式主要用于系统内部多次Measure的情形,并不是真的说你想要多大最后就真有多大)

  • EXACTLY

    父控件已经知道你所需的精确大小,你的最终大小应该就是这么大。

  • AT_MOST

    你的大小不能大于父控件给你指定的size,但具体是多少,得看你自己的实现。

上面的三种模式的区别我们弄清楚了,但是父控件是怎样给它的孩子们构建好测量大小和测量模式的呢?这其中必有蹊跷。好吧,冤有头债有主,我们得去ViewGroup类里去找找看。ViewGroup里提供了一个静态方法getChildMeasureSpec用来获取子控件的测量规格,下面是代码和详细注释:

    /**
     *
     * 目标是将父控件的测量规格和child view的布局参数LayoutParams相结合,得到一个
     * 最可能符合条件的child view的测量规格。  

     * @param spec 父控件的测量规格
     * @p
  • 13
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 10
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值