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

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


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

android:visibility=gone

  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

  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

  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))输入日志信息如下:

  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,会被忽略掉,比如如下代码:

  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,如下代码:

  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参数时报异常?异常信息如下:

  1. java.lang.ClassCastException: android.view.ViewGroup$LayoutParams cannot be cast to android.view.ViewGroup$MarginLayoutParams

java.lang.Object

 ↳ android.view.ViewGroup.LayoutParams
   ↳ 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. 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

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

  1. private float mDownX, mDownY, x, y;
  2. private int dx, dy, il, ir, it, ib;
  3. @Override
  4. public boolean onTouchEvent(MotionEvent event) {
  5. x = event.getX();
  6. y = event.getY();
  7. switch (event.getAction()) {
  8. case MotionEvent.ACTION_DOWN:
  9. mDownX = event.getX();
  10. mDownY = event.getY();
  11. il = getLeft();
  12. ir = getRight();
  13. it = getTop();
  14. ib = getBottom();
  15. break;
  16. case MotionEvent.ACTION_MOVE:
  17. case MotionEvent.ACTION_UP:
  18. dx += Math.round(x - mDownX);
  19. dy += Math.round(y - mDownY);
  20. layout(il + dx, it + dy, ir + dx, ib + dy);
  21. break;
  22. }
  23. return true;
  24. }

深入理解android view 生命周期的更多相关文章

  1. 深入懂得android view 生命周期

    作为自定义 view 的基础,如果不了解android  view 的生命周期 , 那么你将会在后期的维护中发现这样那样的问题 ....... 做过一段时间android 开发的同学都知道,一般 on ...

  2. android view生命周期

    onFinishInflate() 当View中所有的子控件均被映射成xml后触发 onMeasure( int ,  int ) 确定所有子元素的大小 onLayout( boolean ,  in ...

  3. android 的生命周期自我理解

    android的active的生命周期,经过网站的blog学习,加上自己的理解总结如下: 第1种:全新的启动应用程序顺序 onCreate--->onStart---->onResume ...

  4. 理解Fragment的生命周期

    与活动类似,Fragment也有自己的生命周期.理解Fragment的生命周期有助于在Fragment销毁时能恰当地保存其实例,然后在重新创建时能够将其恢复至之前的状态. 下面的“试一试”将研究Fra ...

  5. Android的生命周期学习

    掌握Android的生命周期对于如何一个刚刚接触Android初学者来说是至关重要的,在然后的开发中会给我留有更多的时刻余地,当自己正在认识Android中整个声明周期后,会编写出更加的流畅的程序 应 ...

  6. iOS中View生命周期与布局

    生命周期 说到view的生命周期一般都是指视图控制器的view生命周期. view的创建:loadView 视图控制器(UIViewController)及其子类,无论是手写代码还是storyboar ...

  7. Android 进程生命周期 Process Lifecycle

    Android 进程生命周期 Process Lifecycle 进程的生命周期 Android系统会尽力保持应用的进程,但是有时为了给新的进程和更重要的进程回收一些内存空间,它会移除一些旧的进程. ...

  8. Android组件生命周期(二)

    引言 应用程序组件有一个生命周期——一开始Android实例化他们响应意图,直到结束实例被销毁.在这期间,他们有时候处于激活状态,有时候处于非激活状态:对于活动,对用户有时候可见,有时候不可见.组件生 ...

  9. 理解mpvue的生命周期

    mpvue是美团基于vue开发的一个开发小程序的框架,从而以vue的语法来开发小程序.在生命周期上,mpvue同时支持了vue的生命周期和小程序的生命周期,这可能让新上手的同学费解.这篇文章就来讲讲m ...

随机推荐

  1. UVM基础之----uvm_object

    uvm_void The uvm_void class is the base class for all UVM classes. uvm_object: The uvm_object class ...

  2. sql server 数据库跨服务器备份,复制监视器——快照代理,复制流程

    在做数据库跨服务器复制时,查看复制监视器的快照代理,可以看到复制流程,具体如下: 初始化 连接发布服务器 设置服务器数据库兼容性级别 更新索引的统计信息 在生成快照时锁定已发布的表 复制快照数据(每个 ...

  3. CSS——半透明

    1.opacity:不仅背景半透明,内部其他元素也半透明 2.rgba():只有背景半透明. <!DOCTYPE html> <html lang="en"> ...

  4. dubbo之主机绑定

    主机绑定 查找顺序 缺省主机 IP 查找顺序: 通过 LocalHost.getLocalHost() 获取本机地址. 如果是 127.* 等 loopback 地址,则扫描各网卡,获取网卡 IP. ...

  5. async await 同步方法调用异步方法死锁

    同步方法调用异步方法.GetAwaiter().GetResult()计算函数超时,异步方法所有的回调操作都会期望返回到主线程. 所以会导致各种线程死锁.异步方法中使用ConfigureAwait(f ...

  6. BZOJ 2442: [Usaco2011 Open]修剪草坪 单调队列

    Code: #include<iostream> #include<cstdio> #include<cstring> #include<algorithm& ...

  7. 栈和队列问题:设计一个有 getMin 功能的栈

    [知识点] 栈是一个先进后出(FILO-First In Last Out)的数据结构,队列是一种先进先出(FIFO-First In First Out)的数据结构. [题目] 实现一个特殊的栈,在 ...

  8. Linux内核中_IO,_IOR,_IOW,_IOWR宏的用法与解析

    ref from : http://blog.csdn.net/zhuxiaoping54532/article/details/49680537 main 在驱动程序里, ioctl() 函数上传送 ...

  9. How To:python pip install

    官方网站 https://pypi.python.org/pypi/pip/   下载需要的版本   wget https://pypi.python.org/packages/source/p/pi ...

  10. 16.copy_to定制组合field解决cross-fields搜索弊端

    主要知识点: 在index的mapping中加copy_to字段的方法 copy_to搜索方法     用most_fields策略,去实现cross-fields搜索,有3大弊端,为了解决这三个弊端 ...