VIEW生命周期

View 是在Activity 中使用到的,所以在自定义View的时候,我们需要了解Activity 生命周期方法和View的生命周期方法调用先后顺序。

见如下图

(1) 在Activity onCreate方法中初始化了View , 调用了View 的onFinishInflate

(2) 在执行完 Activity的 onPostResume方法之后(第一次onResume之后)才真正开始了View的绘制工作: onAttachedToWindow --> onMeasure --> 

onSizeChanged --> onLayout --> onDraw

 

当按住锁屏键的时候,Activity和View生命周期发生变化:

Activity onPause之后 调用了 View的 onSaveInstanceState方法

当重新回到页面的时候,Activity和View的生命周变化:

View的生命周期方法没有被调用

 

当点击回退的时候,Activity和View的生命周期发生变化:

Activity   onPause -> onStop -> onDestroy

onDestroy之后 调用了 View的onDetachedFromWindow方法

 

View 的生命周期为

 [改变可见性] --> 构造View --> onFinishInflate --> onAttachedToWindow --> onMeasure -->  onSizeChanged --> onLayout --> onDraw -->

 onDetackedFromWindow

 

 

 

总的可以归结三点:

(1)  在Activity onCreate方法中初始化了View 的时候, 调用了View 的onFinishInflate

(2)  在执行完 Activity的 onResume 方法之后,才真正开始了View的绘制工作:onMeasure -->  onSizeChanged --> onLayout --> onDraw

(3) onMeasure,onSizeChanged,onLayout,onDraw可能由于setVisible或onresume调用多次,而onAttachedToWindow与onDetachedFromWindow在创建与销毁view的过程中只会调用一次

 

 

inflate();

setvisibility();

onMeasure,onLayout,onDraw,最终会直接或间接调用到三个函数,分别为invalidate(),requsetLaytout()以及requestFocus() ,接着

这三个函数最终会调用到ViewRoot中的schedulTraversale()方法,该函数然后发起一个异步消息,消息处理中调用

performTraverser()方法对整个View进行遍历。

 

 

    invalidate()方法 :

 

   说明:请求重绘View树,即draw()过程,假如视图发生大小没有变化就不会调用layout()过程,并且只绘制那些“需要重绘的”

视图,即谁(View的话,只绘制该View ;ViewGroup,则绘制整个ViewGroup)请求invalidate()方法,就绘制该视图。

 

     一般引起invalidate()操作的函数如下:

            1、直接调用invalidate()方法,请求重新draw(),但只会绘制调用者本身。

            2、setSelection()方法 :请求重新draw(),但只会绘制调用者本身。

            3、setVisibility()方法 : 当View可视状态在INVISIBLE转换VISIBLE时,会间接调用invalidate()方法,

                     继而绘制该View。

            4 、setEnabled()方法 : 请求重新draw(),但不会重新绘制任何视图包括该调用者本身。

 

    requestLayout()方法 :会导致调用measure()过程 和 layout()过程 。

 

           说明:只是对View树重新布局layout过程包括measure()和layout()过程,不会调用draw()过程,但不会重新绘制

任何视图包括该调用者本身。

 

    一般引起invalidate()操作的函数如下:

         1、setVisibility()方法:

             当View的可视状态在INVISIBLE/ VISIBLE 转换为GONE状态时,会间接调用requestLayout() 和invalidate方法。

    同时,由于整个个View树大小发生了变化,会请求measure()过程以及draw()过程,同样地,只绘制需要“重新绘制”的视图。

 

    requestFocus()函数说明:

 

          说明:请求View树的draw()过程,但只绘制“需要重绘”的视图。

 

 

 

 

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

做过一段时间android 开发的同学都知道,一般 onXXX 函数都是系统的回调函数。而这篇 blog 也是基于这个思想(或许有点笨)......

 

首先来看三分  创建view 的 日志信息 (自定义View 配置到xml文件中):

android:visibility=gone

[html] view plain copy 

  1. 03-25 19:56:55.934: D/yyyyy(11493): onVisibilityChanged--------=====  
  2. 03-25 19:56:55.934: D/yyyyy(11493): construct 2 parameters .  
  3. 03-25 19:56:55.934: E/yyyyy(11493): onFinishInflate  
  4. 03-25 19:56:55.934: D/yyyyy(11493): onVisibilityChanged--------=====  
  5. 03-25 19:56:55.934: D/yyyyy(11493): onVisibilityChanged--------=====  
  6. 03-25 19:56:55.944: D/yyyyy(11493): onRtlPropertiesChanged--------=====  
  7. 03-25 19:56:55.954: D/yyyyy(11493): onRtlPropertiesChanged--------=====  
  8. 03-25 19:56:55.954: E/yyyyy(11493): onAttachedToWindow  
  9. 03-25 19:56:55.954: D/yyyyy(11493): onWindowVisibilityChanged--------=====  
  10. 03-25 19:56:55.974: D/yyyyy(11493): onWindowFocusChanged--------=====  


android:visibility=invisible

[html] view plain copy 

  1. 03-25 19:57:38.204: D/yyyyy(11694): onVisibilityChanged--------=====  
  2. 03-25 19:57:38.204: D/yyyyy(11694): construct 2 parameters .  
  3. 03-25 19:57:38.204: E/yyyyy(11694): onFinishInflate  
  4. 03-25 19:57:38.204: D/yyyyy(11694): onVisibilityChanged--------=====  
  5. 03-25 19:57:38.204: D/yyyyy(11694): onVisibilityChanged--------=====  
  6. 03-25 19:57:38.224: D/yyyyy(11694): onRtlPropertiesChanged--------=====  
  7. 03-25 19:57:38.224: D/yyyyy(11694): onRtlPropertiesChanged--------=====  
  8. 03-25 19:57:38.224: E/yyyyy(11694): onAttachedToWindow  
  9. 03-25 19:57:38.224: D/yyyyy(11694): onWindowVisibilityChanged--------=====  
  10. 03-25 19:57:38.224: D/yyyyy(11694): onMeasure , width : 1080  ; height: 1557  
  11. 03-25 19:57:38.224: D/yyyyy(11694): onMeasure , width : 144  ; height: 1500  
  12. 03-25 19:57:38.234: D/yyyyy(11694): onSizeChanged  
  13. 03-25 19:57:38.234: I/yyyyy(11694): onLayout --> l: 0  ; r : 144  ; t: 57  ; b: 201  : changed :true  
  14. 03-25 19:57:38.254: D/yyyyy(11694): onMeasure , width : 1080  ; height: 1557  
  15. 03-25 19:57:38.254: D/yyyyy(11694): onMeasure , width : 144  ; height: 1500  
  16. 03-25 19:57:38.254: I/yyyyy(11694): onLayout --> l: 0  ; r : 144  ; t: 57  ; b: 201  : changed :false  
  17. 03-25 19:57:38.264: D/yyyyy(11694): onWindowFocusChanged--------=====  


android:visibility=visible

[html] view plain copy 

  1. 03-25 19:55:15.434: D/yyyyy(11304): construct 2 parameters .  
  2. 03-25 19:55:15.434: E/yyyyy(11304): onFinishInflate  
  3. 03-25 19:55:15.434: D/yyyyy(11304): onVisibilityChanged--------=====  
  4. 03-25 19:55:15.434: D/yyyyy(11304): onVisibilityChanged--------=====  
  5. 03-25 19:55:15.454: D/yyyyy(11304): onRtlPropertiesChanged--------=====  
  6. 03-25 19:55:15.454: D/yyyyy(11304): onRtlPropertiesChanged--------=====  
  7. 03-25 19:55:15.454: E/yyyyy(11304): onAttachedToWindow  
  8. 03-25 19:55:15.454: D/yyyyy(11304): onWindowVisibilityChanged--------=====  
  9. 03-25 19:55:15.454: D/yyyyy(11304): onMeasure , width : 1080  ; height: 1557  
  10. 03-25 19:55:15.454: D/yyyyy(11304): onMeasure , width : 144  ; height: 1500  
  11. 03-25 19:55:15.464: D/yyyyy(11304): onSizeChanged  
  12. 03-25 19:55:15.464: I/yyyyy(11304): onLayout --> l: 0  ; r : 144  ; t: 57  ; b: 201  : changed :true  
  13. 03-25 19:55:15.474: D/yyyyy(11304): onMeasure , width : 1080  ; height: 1557  
  14. 03-25 19:55:15.474: D/yyyyy(11304): onMeasure , width : 144  ; height: 1500  
  15. 03-25 19:55:15.474: I/yyyyy(11304): onLayout --> l: 0  ; r : 144  ; t: 57  ; b: 201  : changed :false  
  16. 03-25 19:55:15.474: D/yyyyy(11304): onDraw--------=====  
  17. 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))输入日志信息如下:

[html] view plain copy 

  1. 03-25 20:37:51.284: E/yyyyy(12530): construct 1 parameter  
  2. 03-25 20:37:51.294: D/yyyyy(12530): onVisibilityChanged--------=====  
  3. 03-25 20:37:51.314: D/yyyyy(12530): onVisibilityChanged--------=====  
  4. 03-25 20:37:51.314: D/yyyyy(12530): onRtlPropertiesChanged--------=====  
  5. 03-25 20:37:51.314: D/yyyyy(12530): onRtlPropertiesChanged--------=====  
  6. 03-25 20:37:51.314: E/yyyyy(12530): onAttachedToWindow  
  7. 03-25 20:37:51.314: D/yyyyy(12530): onWindowVisibilityChanged--------=====  
  8. 03-25 20:37:51.314: D/yyyyy(12530): onMeasure , width : 1080  ; height: 1557  
  9. 03-25 20:37:51.314: D/yyyyy(12530): onSizeChanged  
  10. 03-25 20:37:51.324: I/yyyyy(12530): onLayout --> l: 0  ; r : 1080  ; t: 0  ; b: 1557  : changed :true  
  11. 03-25 20:37:51.324: D/yyyyy(12530): onMeasure , width : 1080  ; height: 1557  
  12. 03-25 20:37:51.324: I/yyyyy(12530): onLayout --> l: 0  ; r : 1080  ; t: 0  ; b: 1557  : changed :false  
  13. 03-25 20:37:51.324: D/yyyyy(12530): onDraw--------=====  
  14. 03-25 20:37:51.344: D/yyyyy(12530): onWindowFocusChanged--------=====  

从测试结果来看,默认情况下view的长和宽默认和父 view 的长和宽一致 。

 

虽然调用了onDraw 函数,但是在屏幕上却看不到任何内容,什么原因?
当看不到任何内容时,请先检查 View要绘制的内容是否制定。

 

为什么我指定了LayoutParameters,却没有效果?

在不恰当的生命周期中指定LayoutParameters,会被忽略掉,比如如下代码:

[java] view plain copy 

  1. @Override  
  2.     protected void onCreate(Bundle savedInstanceState) {  
  3.         super.onCreate(savedInstanceState);  
  4.         // setContentView(R.layout.activity_main);  
  5.         view = new CusView(this);  
  6.         view.setImageResource(R.drawable.ic_launcher);  
  7.         FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(70, 70);  
  8.         view.setLayoutParams(params);  
  9.         setContentView(view);  
  10.     }  

正确的方法应该是放到   onWindowFocusChanged 方法获取到焦点后再指定LayoutParameters,如下代码:

[java] view plain copy 

  1. @Override  
  2.     public void onWindowFocusChanged(boolean hasFocus) {  
  3.         // TODO Auto-generated method stub  
  4.         super.onWindowFocusChanged(hasFocus);  
  5.         if (hasFocus) {  
  6.             view.setImageResource(R.drawable.ic_launcher);  
  7.             FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(70, 70);  
  8.             view.setLayoutParams(params);  
  9.         }  
  10.     }  


为什么我指定LayoutParameters参数时报异常?异常信息如下:

[html] view plain copy 

  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

[html] view plain copy 

  1. 03-25 21:15:35.404: D/yyyyy(14589): onWindowFocusChanged--------=====  
  2. 03-25 21:15:35.484: D/yyyyy(14589): onWindowVisibilityChanged--------=====  
  3. 03-25 21:15:35.504: D/yyyyy(14589): onDetachedFromWindow--------=====  

android:visibility=gone

[html] view plain copy 

  1. 03-25 21:16:09.964: D/yyyyy(14736): onWindowFocusChanged--------=====  
  2. 03-25 21:16:10.054: D/yyyyy(14736): onWindowVisibilityChanged--------=====  
  3. 03-25 21:16:10.064: D/yyyyy(14736): onDetachedFromWindow--------=====  

android:visibility=invisible

[html] view plain copy 

  1. 03-25 21:16:42.534: D/yyyyy(14860): onWindowFocusChanged--------=====  
  2. 03-25 21:16:42.594: D/yyyyy(14860): onWindowVisibilityChanged--------=====  
  3. 03-25 21:16:42.614: D/yyyyy(14860): onDetachedFromWindow--------=====  


从以上内容可以看到,visibility属性对view的销毁流程没有影响。

 

综上所述:View 的关键生命周期为    [改变可见性]  -->   构造View   -->      onFinishInflate  -->   onAttachedToWindow  -->  onMeasure  -->  onSizeChanged  -->  onLayout  -->   onDraw  -->  onDetackedFromWindow

 

最后给出一小段代码用于在屏幕上拖动view(通过修改view的 layout ):

[java] view plain copy 

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;  
    }  

转自:https://blog.csdn.net/jyw935478490/article/details/69397248

 

 

 

 

 

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值