Android View的生命周期详解

View生命周期相关方法:

  1. onFinishInflate() 当View中所有的子控件均被映射成xml后触发 
  2. onMeasure( int ,  int ) 确定所有子元素的大小 
  3. onLayout( boolean ,  int ,  int ,  int ,  int ) 当View分配所有的子元素的大小和位置时触发     
  4. onSizeChanged( int ,  int ,  int ,  int ) 当view的大小发生变化时触发  
  5. onDraw(Canvas) view渲染内容的细节  
  6. onKeyDown( int , KeyEvent) 有按键按下后触发  
  7. onKeyUp( int , KeyEvent) 有按键按下后弹起时触发  
  8. onTrackballEvent(MotionEvent) 轨迹球事件  
  9. onTouchEvent(MotionEvent) 触屏事件  
  10. onFocusChanged( boolean ,  int , Rect) 当View获取或失去焦点时触发   
  11. onWindowFocusChanged( boolean ) 当窗口包含的view获取或失去焦点时触发  
  12. onAttachedToWindow() 当view被附着到一个窗口时触发  
  13. onDetachedFromWindow() 当view离开附着的窗口时触发,Android123提示该方法和  onAttachedToWindow() 是相反的。  
  14. 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;
    }

posted on 2015-11-21 18:34  Jasonxcj  阅读(9489)  评论(0编辑  收藏  举报

导航