文档章节

android view从无到有的过程

abcijkxyz
 abcijkxyz
发布于 2016/07/30 17:23
字数 1774
阅读 0
收藏 0
点赞 0
评论 0

在搜集Android view绘制流程的相关知识时,发现这里面的流程还是有些复杂的,准备了好几天,才敢提起笔来。下面就直入主题吧!
view绘制流程是从ViewRoot的performTraversals()方法中开始的,在该方法中会执行view绘制的三部曲,即:measure(测量视图的大小),layout(确定视图的位置)draw(绘制视图的内容)。下面这张图明确的展示了该过程:

(注:图片来源于工匠若水博客
1、measure的过程
public final void measure(int widthMeasureSpec, int heightMeasureSpec) {  
    ...
    onMeasure(widthMeasureSpec, heightMeasureSpec);  
    ...
}
可以看到该方法是final的,所以不需要子类重写,里面的实现主要就是调用了onMeasure。那么传入的两个参数是什么呢?那就涉及到MeasureSpec了,MeasureSpec由specMode(规格)和specSize(大小)组成,规格有三种,它跟大小对应关系如下:
1. EXACTLY
表示父视图希望子视图的大小应该是由specSize的值来决定的,系统默认会按照这个规则来设置子视图的大小,开发人员当然也可以按照自己的意愿设置成任意的大小。
2. AT_MOST
表示子视图最多只能是specSize中指定的大小,开发人员应该尽可能小得去设置这个视图,并且保证不会超过specSize。系统默认会按照这个规则来设置子视图的大小,开发人员当然也可以按照自己的意愿设置成任意的大小。
3. UNSPECIFIED
表示开发人员可以将视图按照自己的意愿设置成任意的大小,没有任何限制。这种情况比较少见,不太会用到。
对于最外层的根视图,这两个参数是如何确定的呢?原来是调用的getRootMeasureSpec,具体实现如下:
private int getRootMeasureSpec(int windowSize, int rootDimension) {  
    int measureSpec;  
    switch (rootDimension) {  
    case ViewGroup.LayoutParams.MATCH_PARENT:  
        measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);  
        break;  
    case ViewGroup.LayoutParams.WRAP_CONTENT:  
        measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);  
        break;  
    default:  
        measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);  
        break;  
    }  
    return measureSpec;  
}
这个函数传入的参数是窗口大小和MATCH_PARENT,这就是为什么根视图总是铺满屏幕的原因。
再来看看OnMeasure,具体实现如下:
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {  
    setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),  
    getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));  
}
onMeasure里面主要是使用setMeasuredDimension来设置视图的大小,这样就完成了一次measure的过程,当然,一个布局中一般都会包含多个子视图,每个子视图都需要经历一次measure过程。
ViewGroup中定义了一个measureChildren()方法来测量子视图的大小,如下:
protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {  
    final int size = mChildrenCount;  
    final View[] children = mChildren;  
    for (int i = 0; i < size; ++i) {  
        final View child = children[i];  
        if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {  
            measureChild(child, widthMeasureSpec, heightMeasureSpec);  
        }  
    }  
}
里面循环调用了measureChild,其实现为:
protected void measureChild(View child, int parentWidthMeasureSpec,  
    int parentHeightMeasureSpec) {  
       ...
       child.measure(childWidthMeasureSpec, childHeightMeasureSpec);  
}
这里面又调用到了view的measure方法,所以这其实是个递归调用,不断的去测量设置子视图的大小,直至全部测完。
2、layout过程
public void layout(int l, int t, int r, int b) {
    ...
    setFrame(l, t, r, b); 
    ... 
    onLayout(changed, l, t, r, b);
    ...
}
主要是调用了setFrame(用来设置坐标)和onLayout方法,View里面OnLayout是空实现,因为onLayout()过程是为了确定视图在布局中的位置,而这个操作应该是由布局来完成的,即父视图决定子视图的显示位置。而ViewGroup里面的是抽象方法,也就是需要其子类去实现。
以Linearlayout为例,看下这个过程:
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
	if (mOrientation == VERTICAL) {
		layoutVertical(l, t, r, b);
	} else {
		layoutHorizontal(l, t, r, b);
	}
}
void layoutVertical(int left, int top, int right, int bottom) {
	...
	for (int i = 0; i < count; i++) {
		final View child = getVirtualChildAt(i);
		if (child == null) {
			childTop += measureNullChild(i);
		} else if (child.getVisibility() != GONE) {//
			final int childWidth = child.getMeasuredWidth();
			final int childHeight = child.getMeasuredHeight();
			...
			setChildFrame(child, childLeft, childTop + getLocationOffset(child),
			childWidth, childHeight);
		}
	}
}
private void setChildFrame(View child, int left, int top, int width, int height) {
	child.layout(left, top, left + width, top + height);
}
可以看到其实是遍历子view,然后又去调用layout,这样就不停的循环,直到遍历完所有子view。由于view的layout过程中调用了setFrame方法,可以设置视图的大小,就跟measure的功能重合了,所以这里设置的话有可能会使之前measure的计算失效。
3、draw过程
public void draw(Canvas canvas) { 
	... 
	// Step 1, draw the background, if needed   
	int saveCount; 
	if (!dirtyOpaque) { 
		final Drawable background = mBackground;  
		if (background != null) {
			final int scrollX = mScrollX;
			final int scrollY = mScrollY;
			if (mBackgroundSizeChanged) {
				background.setBounds(0, 0,  mRight - mLeft, mBottom - mTop); 
				mBackgroundSizeChanged = false; 
			}
			if ((scrollX | scrollY) == 0) {  
				ckground.draw(canvas);  
			} else {
				canvas.translate(scrollX, scrollY); 
				background.draw(canvas);  
				canvas.translate(-scrollX, -scrollY); 
			} 
		} 
	}
	...
	// Step 3, draw the content  
	if (!dirtyOpaque) onDraw(canvas);  
	...
	// Step 4, draw the children  
	dispatchDraw(canvas);  
	...
	// Step 6, draw decorations (scrollbars)  
	onDrawScrollBars(canvas);  
	return;  
}
这其中最主要的是调用了onDraw和dispatchDraw方法。onDraw是一个空方法,需要子view自己去实现,而ViewGroup的dispatchDraw()方法主要是遍历子view,然后调用drawChild方法,而drawChild又是调用的draw方法,这样就又构成了一个循环调用。
我们经常会使用invalidate和postinvalidate来重绘视图,那这两个函数为什么会有绘图的功能呢?
invalidate里面其实是调用了invalidateChild方法,该方法实现如下:
public final void invalidateChild(View child, final Rect dirty) {
	ViewParent parent = this;
	...
	do {
		......
		//循环层层上级调运,直到ViewRootImpl返回null
		parent = parent.invalidateChildInParent(location, dirty);
		......
	} while (parent != null);
}
当调用到ViewGroup的invalidateChildInparent方法时,只是计算一下需要重绘的矩形区域,直到调用到ViewRoot的该方法。在ViewRoot的invalidateChildInparent里面调用了scheduleTraversals,我们看下该方法的实现:
public void scheduleTraversals() {  
    if (!mTraversalScheduled) {  
        mTraversalScheduled = true;  
        sendEmptyMessage(DO_TRAVERSAL);  
    }  
}
发送了一个message消息,对这个消息的处理是怎样的呢?如下:
public void handleMessage(Message msg) {  
    switch (msg.what) {  
    case DO_TRAVERSAL:  
        if (mProfile) {  
            Debug.startMethodTracing("ViewRoot");  
        }  
        performTraversals();  
        if (mProfile) {  
            Debug.stopMethodTracing();  
            mProfile = false;  
        }  
        break;  
    ......  
}
看到performTraversals了吗?没错,它就是绘制视图的入口函数,上面已经详细分析过了。
postinvalidate是用来在子线程中更新视图的,简单说下这个过程,调用顺序是这样的:postInvalidate-->postInvalidateDelayed-->dispatchInvalidateDelayed-->sendMessageDelayed,然后在handleMessage的处理中(UI线程),又调用了invalidate。
invalidate被调用的地方通常有以下几处:setSelection setVisibility setEnabled requestFocus ,当然我们也可以手动调用来强制更新视图。虽然invalidate最终调用了performTraversals,但是假如视图无需重绘并且发生大小没有变化就不会调用measure和layout过程,并且只绘制那些调用了invalidate()方法的 View。
requestLayout的过程跟invalidate的过程类似,最终也是调用了ViewRoot的performTraversals方法,不过由于设置的标记不同,所以requestLayout()方法会调用measure和layout过程,不会调用draw过程,也不会重新绘制任何View包括该调用者本身。

总结一下:
1、这三个过程都是从上而下,从父到子的,即:先设置父视图,然后遍历子视图,并对其设置。
2、自定义view时,我们可以重写onMeasure(非必须)和onDraw方法,在onMeasure的实现里调用setMeasuredDimension或者super.onMeasure来设置视图大小。
3、自定义ViewGroup时,我们可以重写onLayout(必须)方法,在里面调用view的layout方法设置视图的位置。

© 著作权归作者所有

共有 人打赏支持
abcijkxyz
粉丝 61
博文 6195
码字总数 1876
作品 0
深圳
项目经理
Android动画:献上一份详细 & 全面的动画知识学习攻略

前言 动画的使用 是 开发中常用的知识 可是动画的种类繁多、使用复杂,每当需要 采用自定义动画 实现 复杂的动画效果时,很多开发者就显得束手无策 本文将献上一份动画的全面介绍攻略,包括动...

Carson_Ho ⋅ 06/06 ⋅ 0

Android解析WindowManagerService(三)Window的删除过程

相关文章 Android系统启动系列 Android深入四大组件系列 Android应用进程启动过程系列 Android解析WindowManager系列 前言 在本系列文章中,我提到过:Window的操作分为两大部分,一部分是W...

刘望舒 ⋅ 05/05 ⋅ 0

Android 动画:这是一份详细 & 清晰的 动画学习指南

前言 动画的使用 是 开发中常用的知识 可是动画的种类繁多、使用复杂,每当需要 采用自定义动画 实现 复杂的动画效果时,很多开发者就显得束手无策 本文将献上一份动画的全面介绍攻略,包括动...

Carson_Ho ⋅ 05/03 ⋅ 0

android清理缓存动画、天气APP、购物下单选择器、阅读APP、饿了么demo等源码

Android精选源码 android将文本内容局部变颜色代码(http://www.apkbus.com/thread-597560-1-1.html) Android遮罩对比图效果组件(http://www.apkbus.com/thread-597563-1-1.html) 一个能让你了...

逆鳞龙 ⋅ 04/17 ⋅ 0

Android中View的绘制流程

安卓view的绘制流程主要分为三个阶段: 一、测量过程(mesure) 二、确定子元素的过程(layout) 三、绘制过程(draw) 我们在activity的oncread方法中会使用setContentView来填充一个布局 ...

小怪兽是程序员 ⋅ 06/07 ⋅ 0

Android HookActivity一行代码实现开屏广告

需求背景 多个产品线都需要实现开屏广告,我们产品广告都是接的我们自家广告SDK,而只是几行代码请求我们广告,广告SDK会把View封装好返回来,要做的事情只是获取响应结果,并且出来. 如何实...

xwdz ⋅ 05/09 ⋅ 0

想把GridView添加到ViewPager里面。报空指针异常

用TabLayout和ViewPager关联后,想把GridView添加到ViewPager里面。报空指针异常。不知道哪里写错了 04-12 08:39:37.363 25278-25278/? E/AndroidRuntime: FATAL EXCEPTION: main Process: ...

csl232 ⋅ 04/13 ⋅ 0

Android动画绘制原理(源码解析)

个人博客地址 http://dandanlove.com/ 前言 Android 平台提供了三类动画,一类是 Tween 动画-Animation,即通过对场景里的对象不断做图像变换 ( 平移、缩放、旋转 ) 产生动画效果;第二类是 ...

静默加载 ⋅ 04/30 ⋅ 0

Handler和AsyncTask

在Android中实现异步任务机制有两种方式,Handler和AsyncTask。 Handler模式需要为每一个任务创建一个新的线程,任务完成后通过Handler实例向UI线程发送消息,完成界面的更新,这种方式对于整...

hisense20112784 ⋅ 2017/06/03 ⋅ 0

Android源码剖析之Framework层进阶版(Wms窗口管理)

本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 上一篇我们主要讲了Ams,篇幅有限,本篇再讲讲Wms,即WindowManagerService,管理窗口的服务。主要负责窗口的创建、删除、...

liuzxgeek ⋅ 2016/08/26 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

js模拟栈和队列

栈和队列 栈:LIFO(先进后出)一种数据结构 队列:LILO(先进先出)一种数据结构 使用的js方法 1.push();可以接收任意数量的参数,把它们逐个推进队尾(数组末尾),并返回修改后的数组长度。 2....

LIAOJIN1 ⋅ 11分钟前 ⋅ 0

180619-Yaml文件语法及读写小结

Yaml文件小结 Yaml文件有自己独立的语法,常用作配置文件使用,相比较于xml和json而言,减少很多不必要的标签或者括号,阅读也更加清晰简单;本篇主要介绍下YAML文件的基本语法,以及如何在J...

小灰灰Blog ⋅ 19分钟前 ⋅ 0

IEC60870-5-104规约传送原因

1:周期循环2:背景扫描3:自发4:初始化5:请求6:激活7:激活确认8:停止激活9:停止激活确认10:激活结束11:远程命令引起的返送信息12:当地命令引起的返送信息13:文件传送20:响应总召...

始终初心 ⋅ 32分钟前 ⋅ 0

【图文经典版】冒泡排序

1、可视化排序过程 对{ 6, 5, 3, 1, 8, 7, 2, 4 }进行冒泡排序的可视化动态过程如下 2、代码实现    public void contextLoads() {// 冒泡排序int[] a = { 6, 5, 3, 1, 8, 7, 2, ...

pocher ⋅ 42分钟前 ⋅ 0

ORA-12537 TNS-12560 TNS-00530 ora-609解决

oracle 11g不能连接,卡住,ORA-12537 TNS-12560 TNS-00530 TNS-12502 tns-12505 ora-609 Windows Error: 54: Unknown error 解决方案。 今天折腾了一下午,为了查这个问题。。找了N多方案,...

lanybass ⋅ 56分钟前 ⋅ 0

IDEA反向映射Mybatis

1.首先在pom文件的plugins中添加maven对mybatis-generator插件的支持 ` <!-- mybatis逆向工程 --><plugin><groupId>org.mybatis.generator</groupId><artifactId>mybatis-generator-ma......

lichengyou20 ⋅ 今天 ⋅ 0

4.10/4.11/4.12 lvm讲解 4.13 磁盘故障小案例

准备磁盘分区 fdisk /dev/sdb n 创建三个新分区,分别1G t 改变分区类型为8e 准备物理卷 pvcreate /dev/sdb1 pvcreate /dev/sdb2 pvcreate /dev/sdb3 pvdisplay/pvs 列出当前的物理卷 pvremo...

Linux_老吴 ⋅ 今天 ⋅ 0

zabbix 3.4安装

#已装好lamp环境 1.安装相关yum仓库 rpm -i http://repo.zabbix.com/zabbix/3.4/rhel/7/x86_64/zabbix-release-3.4-2.el7.noarch.rpm #tip:rpm -ql zabbix-release 看上面这个软件装了哪些东......

山月关 ⋅ 今天 ⋅ 0

Java的Excel导出工具类

首先在POM中引入需要的Jar <dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.47</version></dependency><dependency><groupId>o......

Kxvz ⋅ 今天 ⋅ 0

springboot 使用jsp

目录结构: 启动文件的Application必须在contorller文件的父级 文件路径在src/main/webapp下面 我的配置:前缀是/WEB-INF/jsp/ pom.xml需要加入tomcat-embed-jasper, 对jsp的支持的依赖 <de...

夜醒者 ⋅ 今天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部