1.2. Activity、Window、DecorView之间关系
首先来看一下Activity中setContentView源码:
public void setContentView(@LayoutRes int layoutResID) {
//将xml布局传递到Window当中
getWindow().setContentView(layoutResID);
initWindowDecorActionBar();
}
从代码可以看出,Activity
的setContentView
实质是将View
传递到Window
的setContentView()
方法中,Window
的setContenView
会在内部调用installDecor()
方法创建DecorView
,看一下它的部分源码:
public void setContentView(int layoutResID) {
if (mContentParent == null) {
//初始化DecorView以及其内部的content
installDecor();
} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
mContentParent.removeAllViews();
}
if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {
…
} else {
//将contentView加载到DecorVoew当中
mLayoutInflater.inflate(layoutResID, mContentParent);
}
…
}
private void installDecor() {
…
if (mDecor == null) {
//实例化DecorView
mDecor = generateDecor(-1);
…
}
} else {
mDecor.setWindow(this);
}
if (mContentParent == null) {
//获取Content
mContentParent = generateLayout(mDecor);
}
…
}
protected DecorView generateDecor(int featureId) {
…
return new DecorView(context, featureId, this, getAttributes());
}
通过generateDecor()
new一个DecorView
,然后调用generateLayout()
获取DecorView
中content
,最终通过inflate
将Activity
视图添加到DecorView
中的content
中,但此时DecorView
还未被添加到Window
中。添加操作需要借助ViewRootImpl
。
ViewRootImpl
的作用是用来衔接WindowManager
和DecorView
,在Activity
被创建后会通过WindowManager
将DecorView
添加到PhoneWindow
中并且创建ViewRootImpl
实例,随后将DecorView
与ViewRootImpl
进行关联,最终通过执行ViewRootImpl
的performTraversals()
开启整个View树的绘制。
关于Activity在何时将DecorView添加到Window以及何时创建 ViewRootImpl,这块内容牵扯面比较广,涉及到Activity启动流程、ActivityManagerService(AMS)、WindowManagerService(WMS),内容太过于深入加上作者能力有限就不误人子弟了。如有兴趣推荐查阅刘皇叔《Android进阶解密》,书中对这方面内容讲解还是比较全面的 。
2. 绘制过程
从第一小节可知,View的绘制是从ViewRootImpl
的performTraversals()
方法开始,从最顶层的View(ViewGroup)
开始逐层对每个View
进行绘制操作,下面来看一下该方法部分源代码:
private void performTraversals() {
…
//measur过程
performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
…
//layout过程
performLayout(lp, desiredWindowWidth, desiredWindowHeight);
…
//draw过程
performDraw();
}
这方法大概有几百行,机智的作者抽出三句精华呈现给大家~~~
- measure:为测量宽高过程,如果是ViewGroup还要在onMeasure中对所有子View进行measure操作。
- layout:用于摆放View在ViewGroup中的位置,如果是ViewGroup要在onLayout方法中对所有子View进行layout操作。
- draw:往View上绘制图像。
示意图如下: 确实不想画图了,从刚哥的书里拍一张吧~~~
2.1 Measure
performMeasure()
源码
private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
if (mView == null) {
return;
}
try {
mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
} finally {
Trace.traceEnd(Trace.TRACE_TAG_VIEW);
}
}
可以看出从mView(最顶层ViewGroup)开始进行测量操作,然后逐层遍历View并执行measure操作。
MeasureSpac
Measure
是View
绘制三个过程中的第一步,提到Measure
就不得不提MeasureSpac
它是一个32位int
类型数值,高两位SpacMode
代表测量模式,低30位SpacSize
代表测量尺寸,是View的内部类,源码如下:
public class MeasureSpec {
private static final int MODE_SHIFT = 30;
private static final int MODE_MASK = 0x3 << MODE_SHIFT;
public static final int UNSPECIFIED = 0 << MODE_SHIFT;
public static final int EXACTLY = 1 << MODE_SHIFT;
public static final int AT_MOST = 2 << MODE_SHIFT;
}
内部也包含三种测量模式:
-
**UNSPECIFIED :**父布局不会对子View做任何限制,例如我们常用的
ScrollView
就是这种测量模式。 -
**EXACTLY :**精确数值,比如使用了
match_parent
或者xxxdp,表示父布局已经决定了子View
的大小,通常在这种情况下View
的尺寸就是SpacSize
-
**AT_MOST :**自适应,对应
wrap_content
子View可以根据内容设置自己的大小,但前提是不能超出父ViewGroup
的宽高。
注意点:
在我们自定义View的过程中都会在onMeasure中进行宽高的测量,这个方法会从父布局中接收两个参数
widthMeasureSpac
和heightMeasureSpac
,所以子布局的宽高大小需要受限于父布局。
在自定义View宽高测量的过程中,我们需要获取MeasurSpac
中的宽高和测量模式,自定义ViewGroup
也必须给子View传递MeasurSpac
,Android也给我们提供了计算MeasurSpac
和通过MeasurSpac
获取相应值的方式,都位于MeasurSpac
中,具体代码如下:
public static class MeasureSpec {
public static int makeMeas
《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
【docs.qq.com/doc/DSkNLaERkbnFoS0ZF】 完整资料开源分享
ureSpec( int size, int mode) {
if (sUseBrokenMakeMeasureSpec) {
return size + mode;
} else {
return (size & ~MODE_MASK) | (mode & MODE_MASK);
}
}
public static int getMode(int measureSpec) {
//noinspection ResourceType
return (measureSpec & MODE_MASK);
}
public static int getSize(int measureSpec) {
return (measureSpec & ~MODE_MASK)
}
}
从ViewGroup
到View
对尺寸和模式进行了一次封装和拆解,其目的是为了减少对象的创建,避免造成不必要的内存浪费。
LayoutParams
在刚接触Android的时候经常有一个疑问,为什么View设置自己的宽高,还要创建一个xxx.LayoutParams
?前面也提到了,子View的宽高是要受限于父布局的,所以不能通过setWidth
或者setHeight
直接设置宽高的,另外 LayoutParams
的作用不仅如此,比如一个View的父布局是RelativeLayout
,可以通过设置RelativeLayout.LayoutParams
的above
,below
等属性来调整在父布局中的位置。
自定义View宽高测量演示
创建一个类继承View,重写其onMeasure()
方法
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//默认宽
int defaultWidth = 0;
//默认高
int defaultHeight = 0;
setMeasuredDimension(
getDefaultSize(defaultWidth, widthMeasureSpec),
getDefaultSize(defaultHeight, heightMeasureSpec));
}
一般的自定义View中,如果对宽高没有特殊需求可直接通过getDefaultSize()
方法获取,该方法位于View中源码如下:
public static int getDefaultSize(int size, int measureSpec) {
//默认尺寸
int result = size;
//获取测量模式
int specMode = MeasureSpec.getMode(measureSpec);
//获取尺寸
int specSize = MeasureSpec.getSize(measureSpec);
switch (specMode) {
case MeasureSpec.UNSPECIFIED:
result = size;
break;
case MeasureSpec.AT_MOST:
case MeasureSpec.EXACTLY:
result = specSize;
break;
}
return result;
}
从代码分析可知,获取mode
和size
后会分别对三种测量模式进行判断,UNSPECIFIED
使用默认尺寸,而AT_MOST
和EXACTLY
使用父布局给出的测量尺寸。尺寸计算完毕后通过setMeasuredDimension(width,height)
设置最终宽高。
2.2 Layout
performLayout()
部分源码:
【延伸Android必备知识点】
【Android部分高级架构视频学习资源】
**Android精讲视频学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!
**任何市场都是优胜略汰适者生存,只要你技术过硬,到哪里都不存在饱和不饱和的问题,所以重要的还是提升自己。懂得多是自己的加分项 而不是必须项。门槛高了只能证明这个市场在不断成熟化!**另外一千个读者就有一千个哈姆雷特,所以以上只是自己的关键,不喜勿喷!
如果你是卡在缺少学习资源的瓶颈上,那么刚刚好我能帮到你。欢迎关注会持续更新和分享的。
BfvjjWji-1640922859289)]
【Android部分高级架构视频学习资源】
**Android精讲视频学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!
**任何市场都是优胜略汰适者生存,只要你技术过硬,到哪里都不存在饱和不饱和的问题,所以重要的还是提升自己。懂得多是自己的加分项 而不是必须项。门槛高了只能证明这个市场在不断成熟化!**另外一千个读者就有一千个哈姆雷特,所以以上只是自己的关键,不喜勿喷!
如果你是卡在缺少学习资源的瓶颈上,那么刚刚好我能帮到你。欢迎关注会持续更新和分享的。