Android View的生命周期详解
View生命周期相关方法:
- onFinishInflate() 当View中所有的子控件均被映射成xml后触发
- onMeasure( int , int ) 确定所有子元素的大小
- onLayout( boolean , int , int , int , int ) 当View分配所有的子元素的大小和位置时触发
- onSizeChanged( int , int , int , int ) 当view的大小发生变化时触发
- onDraw(Canvas) view渲染内容的细节
- onKeyDown( int , KeyEvent) 有按键按下后触发
- onKeyUp( int , KeyEvent) 有按键按下后弹起时触发
- onTrackballEvent(MotionEvent) 轨迹球事件
- onTouchEvent(MotionEvent) 触屏事件
- onFocusChanged( boolean , int , Rect) 当View获取或失去焦点时触发
- onWindowFocusChanged( boolean ) 当窗口包含的view获取或失去焦点时触发
- onAttachedToWindow() 当view被附着到一个窗口时触发
- onDetachedFromWindow() 当view离开附着的窗口时触发,Android123提示该方法和 onAttachedToWindow() 是相反的。
- onWindowVisibilityChanged( int ) 当窗口中包含的可见的view发生变化时触发
综上所述:View 的关键生命周期为 [改变可见性] --> 构造View --> onFinishInflate --> onAttachedToWindow --> onMeasure --> onSizeChanged --> onLayout --> onDraw --> onDetackedFromWindow
首先来看三分 创建view 的 日志信息 (自定义View 配置到xml文件中):
android:visibility=gone
1
2
3
4
5
6
7
8
9
10
|
03 - 25 19 : 56 : 55.934 : D/yyyyy( 11493 ): onVisibilityChanged--------===== 03 - 25 19 : 56 : 55.934 : D/yyyyy( 11493 ): construct 2 parameters . 03 - 25 19 : 56 : 55.934 : E/yyyyy( 11493 ): onFinishInflate 03 - 25 19 : 56 : 55.934 : D/yyyyy( 11493 ): onVisibilityChanged--------===== 03 - 25 19 : 56 : 55.934 : D/yyyyy( 11493 ): onVisibilityChanged--------===== 03 - 25 19 : 56 : 55.944 : D/yyyyy( 11493 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 56 : 55.954 : D/yyyyy( 11493 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 56 : 55.954 : E/yyyyy( 11493 ): onAttachedToWindow 03 - 25 19 : 56 : 55.954 : D/yyyyy( 11493 ): onWindowVisibilityChanged--------===== 03 - 25 19 : 56 : 55.974 : D/yyyyy( 11493 ): onWindowFocusChanged--------===== |
android:visibility=invisible
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
03 - 25 19 : 57 : 38.204 : D/yyyyy( 11694 ): onVisibilityChanged--------===== 03 - 25 19 : 57 : 38.204 : D/yyyyy( 11694 ): construct 2 parameters . 03 - 25 19 : 57 : 38.204 : E/yyyyy( 11694 ): onFinishInflate 03 - 25 19 : 57 : 38.204 : D/yyyyy( 11694 ): onVisibilityChanged--------===== 03 - 25 19 : 57 : 38.204 : D/yyyyy( 11694 ): onVisibilityChanged--------===== 03 - 25 19 : 57 : 38.224 : D/yyyyy( 11694 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 57 : 38.224 : D/yyyyy( 11694 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 57 : 38.224 : E/yyyyy( 11694 ): onAttachedToWindow 03 - 25 19 : 57 : 38.224 : D/yyyyy( 11694 ): onWindowVisibilityChanged--------===== 03 - 25 19 : 57 : 38.224 : D/yyyyy( 11694 ): onMeasure , width : 1080 ; height: 1557 03 - 25 19 : 57 : 38.224 : D/yyyyy( 11694 ): onMeasure , width : 144 ; height: 1500 03 - 25 19 : 57 : 38.234 : D/yyyyy( 11694 ): onSizeChanged 03 - 25 19 : 57 : 38.234 : I/yyyyy( 11694 ): onLayout --> l: 0 ; r : 144 ; t: 57 ; b: 201 : changed : true 03 - 25 19 : 57 : 38.254 : D/yyyyy( 11694 ): onMeasure , width : 1080 ; height: 1557 03 - 25 19 : 57 : 38.254 : D/yyyyy( 11694 ): onMeasure , width : 144 ; height: 1500 03 - 25 19 : 57 : 38.254 : I/yyyyy( 11694 ): onLayout --> l: 0 ; r : 144 ; t: 57 ; b: 201 : changed : false 03 - 25 19 : 57 : 38.264 : D/yyyyy( 11694 ): onWindowFocusChanged--------===== |
android:visibility=visible
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
03 - 25 19 : 55 : 15.434 : D/yyyyy( 11304 ): construct 2 parameters . 03 - 25 19 : 55 : 15.434 : E/yyyyy( 11304 ): onFinishInflate 03 - 25 19 : 55 : 15.434 : D/yyyyy( 11304 ): onVisibilityChanged--------===== 03 - 25 19 : 55 : 15.434 : D/yyyyy( 11304 ): onVisibilityChanged--------===== 03 - 25 19 : 55 : 15.454 : D/yyyyy( 11304 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 55 : 15.454 : D/yyyyy( 11304 ): onRtlPropertiesChanged--------===== 03 - 25 19 : 55 : 15.454 : E/yyyyy( 11304 ): onAttachedToWindow 03 - 25 19 : 55 : 15.454 : D/yyyyy( 11304 ): onWindowVisibilityChanged--------===== 03 - 25 19 : 55 : 15.454 : D/yyyyy( 11304 ): onMeasure , width : 1080 ; height: 1557 03 - 25 19 : 55 : 15.454 : D/yyyyy( 11304 ): onMeasure , width : 144 ; height: 1500 03 - 25 19 : 55 : 15.464 : D/yyyyy( 11304 ): onSizeChanged 03 - 25 19 : 55 : 15.464 : I/yyyyy( 11304 ): onLayout --> l: 0 ; r : 144 ; t: 57 ; b: 201 : changed : true 03 - 25 19 : 55 : 15.474 : D/yyyyy( 11304 ): onMeasure , width : 1080 ; height: 1557 03 - 25 19 : 55 : 15.474 : D/yyyyy( 11304 ): onMeasure , width : 144 ; height: 1500 03 - 25 19 : 55 : 15.474 : I/yyyyy( 11304 ): onLayout --> l: 0 ; r : 144 ; t: 57 ; b: 201 : changed : false 03 - 25 19 : 55 : 15.474 : D/yyyyy( 11304 ): onDraw--------===== 03 - 25 19 : 55 : 15.484 : D/yyyyy( 11304 ): onWindowFocusChanged--------===== |
1、从中不难看到view 默认为可见的 , 不是默认值时先调用 onVisibilityChanged , 但是此时该view 的任何位置信息都不知道。
2、可见性改变后才是调用带有两个参数的构造函数
3、从xml 文件中 inflate 完成
4、将view 加到 window 中 ( View 是gone 的 ,那么View创建生命周期也就结束 )
5、测量view的长宽 ( onMeasure )
6、定位View 在父View中的位置 ( onLayout )-->(View 是invisible , View 创建生命周期结束)
7、onDraw ( 只有可见的 View 才在 window 中绘制 )
在代码中构造View:
setContentView(new CusView(this))输入日志信息如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
03 - 25 20 : 37 : 51.284 : E/yyyyy( 12530 ): construct 1 parameter 03 - 25 20 : 37 : 51.294 : D/yyyyy( 12530 ): onVisibilityChanged--------===== 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onVisibilityChanged--------===== 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onRtlPropertiesChanged--------===== 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onRtlPropertiesChanged--------===== 03 - 25 20 : 37 : 51.314 : E/yyyyy( 12530 ): onAttachedToWindow 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onWindowVisibilityChanged--------===== 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onMeasure , width : 1080 ; height: 1557 03 - 25 20 : 37 : 51.314 : D/yyyyy( 12530 ): onSizeChanged 03 - 25 20 : 37 : 51.324 : I/yyyyy( 12530 ): onLayout --> l: 0 ; r : 1080 ; t: 0 ; b: 1557 : changed : true 03 - 25 20 : 37 : 51.324 : D/yyyyy( 12530 ): onMeasure , width : 1080 ; height: 1557 03 - 25 20 : 37 : 51.324 : I/yyyyy( 12530 ): onLayout --> l: 0 ; r : 1080 ; t: 0 ; b: 1557 : changed : false 03 - 25 20 : 37 : 51.324 : D/yyyyy( 12530 ): onDraw--------===== 03 - 25 20 : 37 : 51.344 : D/yyyyy( 12530 ): onWindowFocusChanged--------===== |
从测试结果来看,默认情况下view的长和宽默认和父 view 的长和宽一致 。
虽然调用了onDraw 函数,但是在屏幕上却看不到任何内容,什么原因?
当看不到任何内容时,请先检查 View要绘制的内容是否制定。
为什么我指定了LayoutParameters,却没有效果?
在不恰当的生命周期中指定LayoutParameters,会被忽略掉,比如如下代码:
1
2
3
4
5
6
7
8
9
10
|
@Override protected void onCreate(Bundle savedInstanceState) { super .onCreate(savedInstanceState); // setContentView(R.layout.activity_main); view = new CusView( this ); view.setImageResource(R.drawable.ic_launcher); FrameLayout.LayoutParams params = new FrameLayout.LayoutParams( 70 , 70 ); view.setLayoutParams(params); setContentView(view); } |
正确的方法应该是放到 onWindowFocusChanged 方法获取到焦点后再指定LayoutParameters,如下代码:
1
2
3
4
5
6
7
8
9
10
|
@Override public void onWindowFocusChanged( boolean hasFocus) { // TODO Auto-generated method stub super .onWindowFocusChanged(hasFocus); if (hasFocus) { view.setImageResource(R.drawable.ic_launcher); FrameLayout.LayoutParams params = new FrameLayout.LayoutParams( 70 , 70 ); view.setLayoutParams(params); } } |
为什么我指定LayoutParameters参数时报异常?异常信息如下:
1
|
java.lang.ClassCastException: android.view.ViewGroup$LayoutParams cannot be cast to android.view.ViewGroup$MarginLayoutParams |
java.lang.Object | ||
? | android.view.ViewGroup.LayoutParams | |
? | android.view.ViewGroup.MarginLayoutParams |
LayoutParameters的参数类型不对,从上面继承关系可以看到MarginLayoutParameters扩展了ViewGroup的layoutParameters ,将其修改为任意支持margin动作的LayoutParameters 。
接下来我们看三份销毁 View 的日志:
android:visibility=visible
1
2
3
|
03 - 25 21 : 15 : 35.404 : D/yyyyy( 14589 ): onWindowFocusChanged--------===== 03 - 25 21 : 15 : 35.484 : D/yyyyy( 14589 ): onWindowVisibilityChanged--------===== 03 - 25 21 : 15 : 35.504 : D/yyyyy( 14589 ): onDetachedFromWindow--------===== |
android:visibility=gone
1
2
3
|
03 - 25 21 : 16 : 09.964 : D/yyyyy( 14736 ): onWindowFocusChanged--------===== 03 - 25 21 : 16 : 10.054 : D/yyyyy( 14736 ): onWindowVisibilityChanged--------===== 03 - 25 21 : 16 : 10.064 : D/yyyyy( 14736 ): onDetachedFromWindow--------===== |
android:visibility=invisible
1
2
3
|
03 - 25 21 : 16 : 42.534 : D/yyyyy( 14860 ): onWindowFocusChanged--------===== 03 - 25 21 : 16 : 42.594 : D/yyyyy( 14860 ): onWindowVisibilityChanged--------===== 03 - 25 21 : 16 : 42.614 : D/yyyyy( 14860 ): onDetachedFromWindow--------===== |
从以上内容可以看到,visibility属性对view的销毁流程没有影响。
综上所述:View 的关键生命周期为 [改变可见性] --> 构造View --> onFinishInflate --> onAttachedToWindow --> onMeasure --> onSizeChanged --> onLayout --> onDraw --> onDetackedFromWindow
最后给出一小段代码用于在屏幕上拖动view(通过修改view的 layout ):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
private float mDownX, mDownY, x, y; private int dx, dy, il, ir, it, ib; @Override public boolean onTouchEvent(MotionEvent event) { x = event.getX(); y = event.getY(); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: mDownX = event.getX(); mDownY = event.getY(); il = getLeft(); ir = getRight(); it = getTop(); ib = getBottom(); break ; case MotionEvent.ACTION_MOVE: case MotionEvent.ACTION_UP: dx += Math.round(x - mDownX); dy += Math.round(y - mDownY); layout(il + dx, it + dy, ir + dx, ib + dy); break ; } return true ; } |