zoukankan      html  css  js  c++  java
  • 深入懂得android view 生命周期

    作为自定义 view 的基础,如果不了解android  view 的生命周期 , 那么你将会在后期的维护中发现这样那样的问题 .......

    做过一段时间android 开发的同学都知道,一般 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

     

     

    java.lang.Object
     ↳ android.view.ViewGroup.LayoutParams
       ↳ android.view.ViewGroup.MarginLayoutParams

     

    LayoutParameters的参数类型不对,从上面继承关系可以看到MarginLayoutParameters扩展了ViewGroup的layoutParameters ,将其修改为任意支持margin动作的LayoutParameters 。

     

     

    接下来我们看三份销毁 View 的日志:

    android:visibility=visible

     

    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

     

    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

     

    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 ):

    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;
        }
  • 相关阅读:
    48. Rotate Image
    47. Permutations II
    46. Permutations
    45. Jump Game II
    44. Wildcard Matching
    43. Multiply Strings
    42. Trapping Rain Water
    Python_匿名函数
    Python_内置函数之map()
    Python_面向对象_单例模式
  • 原文地址:https://www.cnblogs.com/Free-Thinker/p/5521055.html
Copyright © 2011-2022 走看看