作为自定义 view 的基础,如果不了解 view 的生命周期 , 那么你将会在后期的维护中发现这样那样的问题 .......
做过一段时间 开发的同学都知道,一般 onXXX 函数都是系统的回调函数。而这篇 blog 也是基于这个思想(或许有点笨)......
首先来看三分 创建view 的 日志信息 (自定义View 配置到xml文件中):
android:visibility=gone
- 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
- 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
- 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))输入日志信息如下:
- 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,会被忽略掉,比如如下代码:
- @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,如下代码:
- @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参数时报异常?异常信息如下:
- java.lang.ClassCastException: android.view.ViewGroup$LayoutParams cannot be cast to android.view.ViewGroup$MarginLayoutParams
↳ | android.view.ViewGroup.LayoutParams | |
↳ | onWindowFocusChanged--------=====
android:visibility=gone
android:visibility=invisible
综上所述:View 的关键生命周期为 [改变可见性] --> 构造View --> onFinishInflate --> onAttachedToWindow --> onMeasure --> onSizeChanged --> onLayout --> onDraw --> onDetackedFromWindow 最后给出一小段代码用于在屏幕上拖动view(通过修改view的 layout ):
|