第四章: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有三类:
- UNSPECIFIED:父容器不对View有任何限制,要多大给多大,一般用于系统内部,表示一个测量的状态
- EXACTLY:父容器已经检测出View所需要的精确大小,这个时候View的最终大小就是SpecSize所指定的值.
- 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过程
- 绘制背景background.draw(canvas)
- 绘制自己(onDraw)
- 绘制children(dispatchDraw)
- 绘制装饰(onDrawScrollBars)
4.4 自定义View
4.4.1 自定义View的分类
- 继承View重写onDraw方法:采用这种方式需要自己支持wrap_content并且padding也需要自己处理
- 继承ViewGroup派生特殊的Layout:需要核实地处理ViewGroup的测量,布局这两个过程,并同时处理子元素的测量和布局过程
- 继承特定的View(如TextView):扩展某种已有的View的功能
- 继承特定的ViewGroup(如LinearLayout):像几种view组合在一起,不需要自己处理ViewGroup的测量和布局这两个过程.
4.4.2 自定义View须知
- 让View支持wrap_content
- 如果有必要,让View支持padding:直接继承View的控件,如果不在draw方法中处理padding,那么padding属性是无法起作用的,另外,直接继承自ViewGroup的控件需要在onMeasure和onLayout中考虑padding和子元素的margin对其造成的影响.
- 尽量不要在View中使用Handler:view内部本身就提供了post系列的方法,完全可以替代Handler的作用
- View中如果有线程或者动画,需要及时停止:当包含次View的Activity退出或者当前View被remove时,view的onDetachedFromWindow方法会被调用,和此方法对应的是onAttachToWindow,当包含此View的Activity启动时,View的onAttachedToWindow方法会被调用.
- 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.
第四章:View的工作原理的更多相关文章
- Android艺术开发探索第四章——View的工作原理(下)
Android艺术开发探索第四章--View的工作原理(下) 我们上篇BB了这么多,这篇就多多少少要来点实战了,上篇主席叫我多点自己的理解,那我就多点真诚,少点套路了,老司机,开车吧! 我们这一篇就扯 ...
- Android艺术开发探索第四章——View的工作原理(上)
这章就比较好玩了,主要介绍一下View的工作原理,还有自定义View的实现方法,在Android中,View是一个很重要的角色,简单来说,View是Android中视觉的呈现,在界面上Android提 ...
- 《Android开发艺术探索》读书笔记 (4) 第4章 View的工作原理
本节和<Android群英传>中的第3章Android控件架构与自定义控件详解有关系,建议先阅读该章的总结 第4章 View的工作原理 4.1 初始ViewRoot和DecorView ( ...
- 四、View的工作原理
1.ViewRoot和DecorView ViewRoot对应于ViewRootImpl类,它是连接WindowManager和DecorView的纽带,View的三大流程均是通过ViewRoot来完 ...
- 梳理源码中 View 的工作原理
欢迎Follow我的GitHub, 关注我的掘金. 在View的工作过程中, 执行三大流程完成显示, 测量(measure)流程, 布局(layout)流程, 绘制(draw)流程. 从perform ...
- Android学习笔记View的工作原理
自定义View,也可以称为自定义控件,通过自定义View可以使得控件实现各种定制的效果. 实现自定义View,需要掌握View的底层工作原理,比如View的测量过程.布局流程以及绘制流程,除此之外,还 ...
- View的工作原理(一) 总览View的工作流程
View的工作原理(一) 总览View的工作流程 学习自 <Android开发艺术探索> 简书博主-丶蓝天白云梦 Overview 从本章开始,开始学习View的工作原理,包括View的 ...
- Android之view的工作原理2
学习内容 View的底层工作原理,比如View的测量流程.布局流程以及绘制流程:以及常见的View回调方法:熟悉掌握前面的知识后,自定义View的时候也会更加的得心应手. 4.1 初识ViewRoot ...
- Android 中View的工作原理
Android中的View在Android的知识体系中扮演着重要的角色.简单来说,View就是Android在视觉的体现.我们所展现的页面就是Android提供的GUI库中控件的组合.但是当要求不能满 ...
随机推荐
- Day7-微信小程序实战-引入iconfont(充分利用iconfont图标库的资源)
一.引入iconfont 首先在iconfont.com中注册登陆: 点击上方[图标管理]并进入我的项目 注意:如果没有项目的话,就点击右边的来创建项目 在官网中找到想要的图标之后,以SVG的形式下载 ...
- 设计模式系列之组合模式(Composite Pattern)——树形结构的处理
说明:设计模式系列文章是读刘伟所著<设计模式的艺术之道(软件开发人员内功修炼之道)>一书的阅读笔记.个人感觉这本书讲的不错,有兴趣推荐读一读.详细内容也可以看看此书作者的博客https:/ ...
- redis配置文件中slave-serve-stale-data的解释
redis.conf文件中可以看到slave-serve-stale-data这个参数,作用是什么? 原文解释: # When a slave loses its connection with th ...
- C++的新手入门答疑
基本部分: .ctrl+f5 调试不运行,会出现press anykey to continue f5 调试 .c++变c,修改Stdafx.h,将#include<stdio.h>替换为 ...
- Visual C++线程同步技术剖析:临界区,时间,信号量,互斥量
使线程同步 在程序中使用多线程时,一般很少有多个线程能在其生命期内进行完全独立的操作.更多的情况是一些线程进行某些处理操作,而其他的线程必须对其处理结果进行了解.正常情况下对这种处理结果的了解应当在其 ...
- 网络基础和 TCP、IP 协议
1.网络基本概念 1.1 什么是网络:一些网络设备按照一定的通讯规则(网络协议)进行通讯的系统. 1.2 VPN(虚拟私有网络)加密,相当于专线,从分支机构到总部. 1.3 资源共享的功能和特点: 数 ...
- tkinter的三种几何布局管理类
1.pack() 主要采用块的方式组织子组件 如下: import tkinter root=tkinter.Tk() #创建窗口对象 label=tkinter.Label(root,text='h ...
- sorted 函数及小练习
python 中sorted函数 sorted() 函数对所有可迭代的对象进行排序操作. sorted 语法: sorted(iterable[, cmp[, key[, reverse]]]) 参数 ...
- JDK8 日期格式化
SpringBoot 是为了简化 Spring 应用的创建.运行.调试.部署等一系列问题而诞生的产物,自动装配的特性让我们可以更好的关注业务本身而不是外部的XML配置,我们只需遵循规范,引入相关的依赖 ...
- debian10 安装 详解
准备 下载debian 下载页面,有3个iso,选择第一个,其他两个软件包,我们不需要. 制作启动盘 debian官方提供了一个工具,可以轻松制作启动盘,也可以用opensuse官方提供的一个工具ru ...