ViewRoot对应于ViewRootImpl类,是链接WindowManager和DecorView的纽带。View的三大流程均是经过ViewRoot来完成的。在ActivityThread中,当Activity对象被建立完毕后,会将DecorView添加到Window中,同时会建立ViewRootImpl对象,并将ViewRootImpl对象和DecorView创建关联。android
View的绘制流程从ViewRoot的performTraversals开始,通过measure、layout和draw三个过程才能够把一个View绘制出来,其中:canvas
performTraversals会依次调用performMeasure、performLayout和performDraw三个方法,这三个方法分别完成顶级View的measure、layout和draw这三大流程。其中performMeasure中会调用measure方法,在measure方法中又会调用onMeasure方法,在onMeasure方法中则会对全部子元素进行measure过程,这样就完成了一次measure过程;子元素会重复父容器的measure过程,如此反复完成了整个View数的遍历。另外两个过程相似,大体调用流程以下图:bash
//获取内容栏
ViewGroup content = findViewById(R.android.id.content);
//获取咱们设置的Viewcontext.getChildAt(0);
复制代码
DecorView实际上是一个FrameLayout,View层的事件都先通过DecorView,而后才传给咱们的View。ide
MeasureSpec很大程度上决定一个View的尺寸规格,测量过程当中,系统会将View的layoutParams根据父容器所施加的规则转换成对应的MeasureSpec,再根据这个measureSpec来测量出View的宽/高。oop
MeasureSpec表明一个32位的int值,高2位为SpecMode,低30位为SpecSize,SpecMode是指测量模式,SpecSize是指在某种测量模式下的规格大小。布局
MpecMode有三类;post
1.UNSPECIFIED 父容器不对View进行任何限制,要多大给多大,通常用于系统内部 2.EXACTLY 父容器检测到View所须要的精确大小,这时候View的最终大小就是SpecSize所指定的值,对应LayoutParams中的match_parent和具体数值这两种模式。 3.AT_MOST 父容器指定了一个可用大小即SpecSize,View的大小不能大于这个值,不一样View实现不一样,对应LayoutParams中的wrap_content。优化
当View采用固定宽/高的时候,无论父容器的MeasureSpec的是什么,View的MeasureSpec都是精确模式而且其大小遵循Layoutparams的大小。动画
当View的宽/高是match_parent时,若是他的父容器的模式是精确模式,那View也是精确模式而且大小是父容器的剩余空间;若是父容器是最大模式,那么View也是最大模式而且起大小不会超过父容器的剩余空间。spa
当View的宽/高是wrap_content时,无论父容器的模式是精确仍是最大化,View的模式老是最大化而且不能超过父容器的剩余空间。
自定义一个ListView重写onMeasure方法
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//从新设置高度
heightMeasureSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
复制代码
MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);这个方法就是根据传入的大小和模式生成一个MeasureSpec类型的32位int值。用两位来表示模式,剩下30位表示大小。 因此传入的Integer.MAX_VALUE >> 2就是30位的最大值,模式是MeasureSpec.AT_MOST即表示子视图最多只能是specSize中指定的大小,最大不超过这个大小,至关于warp_content,不超过最大size的效果。
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),
getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));}
复制代码
步骤一:调用onMeasure方法;判断布局是水平仍是竖直;根据布局方向选择measureVertical或者measureHorizontal方法; 步骤二:进入measureVertical方法,遍历子元素并对每个子元素执行measureChildBeforeLayout方法,这个方法内部会调用子元素的measure方法;mTotalLength这个变量来存储LinearLayout在竖直方向上的高度。 步骤三:当子元素测量完毕以后,LinearLayout会根据子元素的状况来测量本身的大小,若高度采用的是match_parent或者具体值,那么他的绘制过程和View一致,若采用warp_content,那么它的高度是全部的子元素所占用的高度+竖直方向上的Padding。
由于View的measure过程和Activity的生命周期不是同步进行,若是View尚未测量完毕,那么获取到的宽/高就是0;因此在Activity的onCreate、onStart、onResume中均没法正确的获取到View的宽/高信息。下面给出4种解决方法。
int widthMeasureSpec = MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec(100,MeasureSpec.EXACTLY);
view.measure(widthMeasureSpec,heightMeasureSpec);
复制代码
(3). wrap_content:
int widthMeasureSpec = MeasureSpec.makeMeasureSpec((1<<30)-1,MeasureSpec.AT_MOST);
int heightMeasureSpec = MeasureSpec.makeMeasureSpec((1<<30)-1,MeasureSpec.AT_MOST);
view.measure(widthMeasureSpec,heightMeasureSpec);
复制代码
在View的默认实现中,View的测量宽高和最终宽高相等,只不过测量宽高造成于measure过程,最终宽高造成于layout过程。但重写view的layout方法可使他们不相等。
经过 onDraw 方法来实现一些不规则的效果,这种效果不方便经过布局的组合方式来达到。这种方式须要本身支持 wrap_content ,而且padding也要去进行处理。
实现自定义的布局方式,须要合适地处理ViewGroup的测量、布局这两个过程,并同时处理子View的测量和布局过程。
扩展某种已有的控件的功能,比较简单,不须要本身去管理 wrap_content 和padding。
比较常见,实现几种view组合一块儿的效果。