第四章:View的工作原理

    4.1 ViewRoot和DecorView

        ViewRoot对应于ViewRootImplement类,它是连接WindowManager和DecorView的纽带,View的三大流程均是通过ViewRoot来完成的.

        在ActivityThread中,当Activity对象被创建完毕后,会将DecorView添加到Window中,同时会创建ViewRootImpl对象,并将ViewRootImpl对象和DecorView建立关联

        View的绘制流程是从ViewRoot的performTraversals方法开始的,它经过measure,layout和draw三个过程来将一个View绘制出来

    4.2 MeasureSpec

        MeasureSpec参与了View的measure过程,在很大程度上决定了一个view的尺寸规格.

      4.2.1 概念

        MeasureSpec代表是一个32位int值,高2位代表SpecMode,低30位代表SpecSize,SpecMode是指测量模式,而SpecSize是指在某种测量模式下的规格大小.

        SpecMode有三类:

      1. UNSPECIFIED:父容器不对View有任何限制,要多大给多大,一般用于系统内部,表示一个测量的状态
      2. EXACTLY:父容器已经检测出View所需要的精确大小,这个时候View的最终大小就是SpecSize所指定的值.
      3. AT_MOST:父容器指定一个可用大小即SpecSize,View的大小不能大于这个值.

      4.2.2 MeasureSpec和LayoutParams的对应关系

          在View测量的时候,系统会将LayoutParams在父容器的约束下转换成对应的MeasureSpec,然后再根据这个MeasureSpec来确定View测量后的宽/高.

    4.3 View的工作流程

        View的工作流程主要是指measure,layout,draw这三大流程,即测量,布局和绘制,其中measure确定View的测量宽/高,layout确定View的最终宽高和四个顶点的位置,而draw则将View绘制到屏幕.

      4.3.1 measure过程

        1. View的measure过程

          View的measure过程由其measure方法来完成,measure方法是一个final类型的方法,子类不能重写此方法,在View的measure方法中会去调用View的onMeasure方法

        2. ViewGroup的measure过程

          除了完成自己的measure过程之外,还会遍历去调用所有子元素的measure方法,各个子元素再递归去执行这个过程.

          ViewGroup是一个抽象类,因此它没有重写View的onMeasure方法,但是提供了measureChildren的方法.

          measureChild的思想就是取出子元素的LayoutParams,然后再通过getChildMeasureSpec来创建子元素的MeasureSpec,接着将MeasureSpec直接传递给View的measure方法进行测量.

          如果它的布局中高度采用的是match_parent或者具体数值,那么它的测量过程和View一致,即高度为SpecSize;如果它的布局中高度采用的是wrap_content,那么它的高度是所有子元素所占用的高度综合,但是仍然不能超过它的父容器的剩余空间.

          measure完成之后,可以通过getMeasuredWight/Height方法就可以正确的取到View的测量宽/高.

            onWindowFocusChanged方法:View已经初始化完毕,宽/高已经准备好了,这个时候去获取宽/高

            view.post(runnable)方法:通过post可以将一个runnable投递到消息队列的尾部,然后等待Looper调用次runnable的时候,View也已经初始化好了.

            ViewTreeObserver:可以使用OnGlobalLayoutListener这个接口,当View树的状态发生改变或者View树内部的View的可见性发生改变,onGlobalLayout方法方法将被回调.

            view.measure:通过手动对View进行measure来得到View的宽/高

      4.3.2 layout过程

          Layout的作用是ViewGroup用来确定子元素的位置,当ViewGroup的位置被确定后,它在onLayout中会遍历所有的子元素并调用其layout方法,在layout方法中onLayout方法又会被调用.

          layout方法确定View本身的位置,而onLayout方法则会确定所有子元素的位置

          layout方法大致流程:首先会通过setFrame方法来设定View的四个顶点的位置,接着调用onLayout方法,用途是父容器确定子元素的位置,和onMeasure方法类似

      4.3.3 draw过程

        1. 绘制背景background.draw(canvas)
        2. 绘制自己(onDraw)
        3. 绘制children(dispatchDraw)
        4. 绘制装饰(onDrawScrollBars)

    4.4 自定义View

        4.4.1 自定义View的分类

        1. 继承View重写onDraw方法:采用这种方式需要自己支持wrap_content并且padding也需要自己处理
        2. 继承ViewGroup派生特殊的Layout:需要核实地处理ViewGroup的测量,布局这两个过程,并同时处理子元素的测量和布局过程
        3. 继承特定的View(如TextView):扩展某种已有的View的功能
        4. 继承特定的ViewGroup(如LinearLayout):像几种view组合在一起,不需要自己处理ViewGroup的测量和布局这两个过程.

        4.4.2 自定义View须知

        1. 让View支持wrap_content
        2. 如果有必要,让View支持padding:直接继承View的控件,如果不在draw方法中处理padding,那么padding属性是无法起作用的,另外,直接继承自ViewGroup的控件需要在onMeasure和onLayout中考虑padding和子元素的margin对其造成的影响.
        3. 尽量不要在View中使用Handler:view内部本身就提供了post系列的方法,完全可以替代Handler的作用
        4. View中如果有线程或者动画,需要及时停止:当包含次View的Activity退出或者当前View被remove时,view的onDetachedFromWindow方法会被调用,和此方法对应的是onAttachToWindow,当包含此View的Activity启动时,View的onAttachedToWindow方法会被调用.
        5. View带有滑动嵌套情形时,需要处理好滑动冲突.

        4.4.3 自定义View

            自定义属性:

            第一步:在values目录下面创建自定义属性的XML(一般以attrs_开头的文件名)

<?xml version="1.0" encoding="UTF-8"?>
<resources>
    <declare-styleable name="CircleView">
        <attr name="circle_color" format="color"/>//定义一个格式为"color"的属性
    </declare-styleable>
</resources>

            第二步:在View的构造方法中解析自定义属性的值并做相应处理.

public CircleView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    TypeArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleView);
    mColor = a.getColor(styleable.CircleView_circle_color, Color.RED);
    a.recycle();
    init();
}加载自定义属性集合CircleView,接着解析CircleView属性集合中的circle_color属性如果在使用时没有指定circle_color这个属性,那么就会选择红色作为默认的颜色值.解析完成之后通过recycle方法来实现资源

            第三步:在布局文件中使用自定义属性

<LinearLayout xmlns:android="http://schemes.android.com/apk/res/android"
    xmlns:app="http://schemes.android.com/apk/res-auto"//必须在布局文件中添加schemas声明
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#ffffff"
    android:orientation="vertical">

    <com.example.CircleView
        android:id="@+id/circleView1"
        android:layout_height="wrap_content"
        android:layout_height="100dp"
        android:layout_margin="20dp"
        app:circle_color="@color/light_green"
        android:padding="20dp"
        android:background="#00000"/>

</LinearLayout>

        其它的View继承也是基于这种思路,measure,layout,draw.

                

时间: 2024-10-11 22:30:17

第四章:View的工作原理的相关文章

Android艺术开发探索第四章——View的工作原理(下)

Android艺术开发探索第四章--View的工作原理(下) 我们上篇BB了这么多,这篇就多多少少要来点实战了,上篇主席叫我多点自己的理解,那我就多点真诚,少点套路了,老司机,开车吧! 我们这一篇就扯一个内容,那就是自定义View 自定义View 自定义View的分类 自定义View的须知 自定义View的实例 自定义View的思想 一.自定义View的分类 自定义View百花齐放,没有什么具体的分类,不过可以从特性大致的分为4类,其实在我看来,就三类,继承原生View,继承View和继承Vie

Android开发艺术探索——第四章View的工作原理

Android开发艺术探索--第四章View的工作原理 4.1 (一)初识ViewToot和DecorView 基本概念 ViewRoot对应于ViewRootImpl类,是连接WindowManager和DecorView的纽带,View的三大流程均是通过ViewRoot来完成的.在ActivityThread中,当Activity对象被创建完成后,会将DecorView添加到View中.同时,会创建ViewRootImpl对象,并将ViewTootImpl对象和DecorView建立关联.

Android艺术开发探索第四章——View的工作原理(上)

这章就比较好玩了,主要介绍一下View的工作原理,还有自定义View的实现方法,在Android中,View是一个很重要的角色,简单来说,View是Android中视觉的呈现,在界面上Android提供了一套完整的GUI库,里面有很多控件,但是有时候往往并不能满足于需求,所以只有自定义View了,我们会简单的说下流程,然后再去实践除了View的三大流程之外,View常见的回调方法也是必须掌握的,比如构造方法,onAttach,onVisibilityChanged,onDetach,另外对于一些

【读书笔记】【Android 开发艺术探索】第4章 View 的工作原理

一.基础知识 1.ViewRoot 和 DecorView ViewRoot 对应 ViewRootImpl 类,它是连接 WindowManager 和 DecorView 的纽带,View 的三大流程都是通过 ViewRoot 来完成的.在ActivityThread 中,当 Activity 对象被创建完毕后,会将 DecorView 添加到 Window 中,同时会创建 ViewRoot 对象. DecorView 添加到窗口 Window 的过程. 图片来自https://yq.ali

Android学习笔记View的工作原理

自定义View,也可以称为自定义控件,通过自定义View可以使得控件实现各种定制的效果. 实现自定义View,需要掌握View的底层工作原理,比如View的测量过程.布局流程以及绘制流程,除此之外,还需要掌握View常见的回调方法.而对于那些具有滑动效果的自定义View,我们还需要处理View的滑动,如果遇到滑动冲突则需要处理相应的滑动冲突. 下面是View的常见回调方法: 构造方法 onAttach onVisibilityChanged onDetach onFinishInflate on

puppet安装使用教程(四)--puppet的工作原理及工作过程

在使用任何软件前我们都需要了解其工作原理,否则会给后续使用带来诸多不便.Puppet采用了非常简单的C/S架构,所有数据的交互都通过SSL进行,以保证安全.它的工作流程如图1所示. 图1 Puppet工作流程 1. 客户端Puppetd向Master发起认证请求,或使用带签名的证书. 2. Master告诉Client你是合法的. 3. 客户端Puppetd调用Facter,Facter探测出主机的一些变量,例如主机名.内存大小.IP地址等.Puppetd将这些信息通过SSL连接发送到服务器端.

第二章:JMeter 工作原理

JMeter的工作原理: 以web性能测试为例, 1.通过 JMeter 来捕获 浏览器的请求 和 Web服务器的响应,捕获到的结果也就是 性能测试脚本 2.通过 性能测试脚本 , JMeter 就可以通过线程组来模拟真实的用户对 Web服务器的 进行压力测试.

jsp基础学习(四)----jsp引擎工作原理

JSP引擎的工作原理 当一个JSP页面第一次被访问的时候,JSP引擎将执行以下步骤:      (1)将JSP页面翻译成一个Servlet,就是一个Java文件,同时也是一个完整的Java程序.(相当于是c语言程序的预处理,补充完整所有的代码)      (2)JSP引擎调用Java编译器对这个Servlet进行编译,得到可执行文件class.(和C语言程序的编译是一样的) (3)JSP引擎调用java虚拟机来解释执行class文件,生成向客户端发送的应答,然后发送给客户端.(jsp是调用jav

第十一章以太网交换机工作原理

在局域网中,交换机是非常重要的网络设备,负责在主机之间快速转发数据帧.交换机与集线器的不同之处在于,交换机工作在数据链路层,能够根据数据帧中的MAC地址进行转发.本章介绍了共享式以太网和交换式以太网的区别,最后重点讲述了交换机进行MAC地址学习以构建MAC地址表的过程,对数据帧的转发原理. 共享式与交换式以太网 共享式以太网 Hub与同轴电缆都是典型的共享式以太网所使用的设备,工作在OSI模型的物理层.Hub和同轴电缆所连接的设备位于一个冲突域中,域中的设备共享带宽,设备间利用CSMA/CD机制