对于android view的测量,我们一句话总结为:"给我位置和大小,我就知道您长到那里"。
为了让大家更好的理解这个结论,我这里先讲一个日常生活中的小故事:不知道大家玩过"瞎子画画"的游戏没,一个人蒙上眼睛,拿笔去画板上画一些指定的图案,另外一个人则充当他的"眼睛",通过语言告诉他在画板那个位置画一个多大的图案。倘若,这个人不告诉那个蒙着眼睛的人,在那个画一个多大的图案。那么这个蒙着眼睛的人此时真是"河里赶大车----------没辙"。其实,android就是这个蒙着眼睛的人,我们必须精确地告诉他如何去画,它才能画出你所想要的图形。
大家是不是对android布局的测量进行现实世界进行类比了。为了实现view具体布局在哪儿,android设计了一个短小精悍又功能强大的类——measurespec类。这样妈妈再也不用担心我不会测量view了。那么,measurespec到底是个什么鬼了。measurespec,归根结底是一个32位的int值。其中高2位表示测量的模式,低30位表示测量view的大小。这样做有什么好处。这样做通过位运算来提高运行效率。
要了解measurespec这个类的来弄去脉的话,务必要对测量的三种模式了解。
1.exactly(精准的)
当您设置view的layout_height属性或layout_width属性为确定的值或者为match_parent(填充父容器)时候,系统就将view测量模式设置为exactly模式。
2.at_most(最大值)
即布局为最大值模式,那么什么时候系统会将view调整为at_most模式了,即当您设置view的layout_height属性或layout_width属性为wrap_content(包裹内容)时候。
3.unspecified(未确定)
即没有确定,没有指定大小测量模式,view即“心有多大,舞台就有多大"。这个方法,一般在自定义控件中才能用到。
view测量的时候,默认是exactly模式,也许你会感到纳闷,textview,edittext这些控件,他怎么就支持wrap_content属性了,难道他重写onmeasure方法,是的,他们都重写onmeasure方法。这就是为什么我们在自定义控件的时候,如果要布局支持wrap_content属性,就需要重写onmeasure方法,来指定wrap_content为确切的大小。
这个关于测量模式的思维导图应该是这样的:
我们知道这么多理论的知识,是不是觉得即枯燥乏味又觉得然并卵。好吧,我们就直接上代码,在代码中解释measurespec如何获取测量模式和测量的大小。源代码如下:
java代码如下:
1
2
3
4
5
6
|
public class myview extends view { public myview(context context, attributeset attrs) { super (context, attrs); } } |
xml代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
<linearlayout xmlns:android= "http://schemas.android.com/apk/res/android" xmlns:tools= "http://schemas.android.com/tools" android:layout_width= "match_parent" android:layout_height= "match_parent" android:orientation= "vertical" tools:context= ".mainactivity" > <com.example.test.myview android:layout_width= "wrap_content" android:layout_height= "wrap_content" android:background= "#00ff00" /> </linearlayout> |
运行效果如下所示:
通过这个短小精悍的例子,充分证明这样一个结论:view测量的时候,默认是exactly模式,你不重写onmeasure方法,即使设置wrap_content属性,他也是填充父容器。
那么,就通过measurespec这个万金油类来重写一下onmeasure方法。相应源代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
public myview(context context, attributeset attrs) { super (context, attrs); } @override protected void onmeasure( int widthmeasurespec, int heightmeasurespec) { setmeasureddimension(measurewidth(widthmeasurespec), measurewidth(heightmeasurespec)); } public int measurewidth( int measurespec) { int result = 0 ; int specmode = measurespec.getmode(measurespec); int specsize = measurespec.getsize(measurespec); if (specmode == measurespec.exactly) { result = specsize; } else { result = 200 ; if (specmode == measurespec.at_most) { result = math.min(specsize, result); } } return result; } |
运行效果如下:
同样的例子,我们只不过是重写了onmeasure方法,通过measurespec.getmode(measurespec)获取测量模式的时候,通过measurespec.getsize(measurespec)获取控件尺寸。判断当布局属性为wrap_content,指定为一确切值,这时,控件就符合wrap_content属性。
以上就是本文的全部内容,希望对大家的学习有所帮助。