由浅入深的说明Android绘图机制

  1. Android中每一个组件的绘制过程,都要经过三个阶段:测量、布局、绘制,分别对应着方法onMeasureonLayoutonDraw(这三个方法定义于View类中)。当然,这三个方法都是允许组件自己重定义的方法,来实现组件对自己的尺寸进行测量(①有child views时,要分别对子组件调用相关测量方法,比如measureChildmeasureChildWithMargins等,并根据子组件的dimension来确定自己的尺寸;②最后要调用setMeasuredDimension(measuredWidth, measuredHeight)方法来保存自己的尺寸信息。)、对自己进行布局(调用onLayout,如果是container,同样需要对子组件进行布局,调用子组件的layout(int l, int t, int r, int b)方法,最终同样会执行子组件的onLayout方法来实现对子组件的布局(布局说白了就是确定自己的绘制位置与大小,即左上右下四个坐标(这样也就提供了大小)),对于容器来说,将其中的具体组件布局好了,也就将其布局好了)以及绘制自己的内容(调用onDraw方法,使用该方法的参数Canvas进行内容绘制);
  2. 从代码中看,Android的绘图是从ViewRootImpl类的performTraversals方法开始的,可以把这个方法视为一个顶层的控制方法,在其中控制整个绘图的流程。具体情况如下所述:
    Ⅰ首先,在其中会调用performMeasure方法,在performMeasure方法中调用View的measure方法(View的measure是final方法,方法原型为:
    public final void measure(int widthMeasureSpec, int heightMeasureSpec)
    也就是说不允许子类修改测量的框架,只能够修改真正进行测量工作的onMeasure方法),进而调用具体组件所实现的onMeasure方法。
    Ⅱ然后,测量结束后会调用performLayout方法,在performLayout方法中调用View的layout方法,该方法原型为:
    public void layout(int l, int t, int r, int b)
    在该方法中会调用View的onLayout方法,对组件进行布局。
    在拓展ViewGroup类的时候,对于所重写的onLayout方法,一般最后一步就是分别调用组件各自的layout方法来“Place the child.”。
    Ⅲ之后,会调用performDraw方法,通过performDraw -> draw -> drawSoftware最终会调用View的draw(Canvas)方法。在draw方法中会有六步操作,在第三步“draw the content”时会调用onDraw(Canvas)方法,进行内容的绘制。
  3. 总结:
    如果我们要拓展ViewGroup类实现一个布局,就要在其中重写onMeasure方法来对布局中的组件进行测量,并在获得其中所有组件的尺寸后计算得到布局的尺寸,然后调用setMeasuredDimension方法进行设置;之后还需要重写onLayout方法,在其中调用各个组件的layout方法,传入计算出的组件坐标位置,实现对组件的布局。
    至于绘制,则由具体的组件自己重写onDraw方法进行实现,在ViewRootImpl类的performTraversals逻辑中进行控制。
  4. 附加:
    ①在测量结束后(调用方法setMeasuredDimension后),就可以调用getMeasuredWidthgetMeasuredHeight来获取视图测量出的宽和高了,在这之前这两个方法返回值均为0;
    在布局结束后,就可以调用方法getWidthgetHeight来获取视图的宽高了。
    以上就是getMeasuredXXXgetXXX的区别,由于一般情况下,会根据测量的情况去布局组件,所以这两个方法的返回值是一样的。
    ②关于MeasureSpec
    MeasureSpec由specSize与specMode两部分组成,前者是大小信息,后者是模式信息。包括的模式有:
    EXACTLY
    表示父视图希望子视图的大小应该是由specSize的值来决定的,系统默认会按照这个规则来设置子视图的大小,开发人员当然也可以按照自己的意愿设置成任意的大小。
    AT_MOST
    表示子视图最多只能是specSize中指定的大小,开发人员应该尽可能小得去设置这个视图,并且保证不会超过specSize。系统默认会按照这个规则来设置子视图的大小,开发人员当然也可以按照自己的意愿设置成任意的大小。
    UNSPECIFIED
    表示开发人员可以将视图按照自己的意愿设置成任意的大小,没有任何限制。这种情况比较少见,不太会用到。
    MeasureSpec是一个32位的int数,其中的2位用来表示模式,余下30位用来表示size。
  5. performTraversals代码时会发现,比如进行布局的时候,代码中所调用的是performLayout方法,在该方法中所执行的关键操作是
    host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());
    当时我想,这一个调用怎么实现对所有组件的布局的呢?还因此在代码中找了好一会儿循环语句,现在我明白了:这个host是个布局对象,调用其layout会进一步调用其onLayout方法,在onLayout方法中实现对所有子组件的遍历布局。同理,测量也是这样的。

更多相关文章

  1. Android模块化和组件化开发简单理解(一)
  2. Android:FlexboxLayout你值得拥有的流布局助力
  3. Android平台OpenMax多媒体引擎介绍
  4. 浅谈Android布局样式
  5. Android艺术开发探索第三章——View的事件体系(上)
  6. (原创)Android入门教程(十五)之-- Activity生命周期及其配置使用
  7. Android(安卓)入门第四讲04-小结-RecyclerView(回顾)+Context(介绍
  8. Android的Intent Filter详解
  9. Android中Handler问题汇总

随机推荐

  1. Android(安卓)仿微信Activity左右切换 【
  2. Android色彩空间像素格式定义及设定
  3. android textview 竖排显示
  4. Android(安卓)NDK纯C++开发
  5. Mac下的android studio快捷键
  6. Android(安卓)使用cmwap访问互联网的办法
  7. 使用Android提供的模拟任意地理位置,报jav
  8. Android访问WebService
  9. android 设置背景透明 半透明 透明
  10. 自制android1.5的源码包