博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
View.requestLayout
阅读量:6194 次
发布时间:2019-06-21

本文共 3771 字,大约阅读时间需要 12 分钟。

 

参考:

 

一、invalidate和postInvalidate

  这两个方法都是在重绘当前控件的时候调用的。invalidate在UI线程中调用,postInvalidate在非UI线程中调用。因为android的UI线程是非线程安全的,所以在非UI线程中,需要使用postInvalidate来使View重绘。当然,Handler和invalidate配合使用,也可以达到目的,但是程序会比较繁琐。

 

二、requestLayout

  如果View对应的位置已经发生了变化,原来被他覆盖的内容需要被显示出来。这时,如果只重新绘制当前View,已经起不到效果。这就需要requestLayout来实现重绘当前View和父View,甚至更上层的View,以此类推。

 

 

下面附上上面博客的关于requestLayout的调用过程的内容:

当一个View调用requestLayout的时候,会给当前的View设置一个FORCE_LAYOUT标记。由此向ViewParent请求布局。这样从这个View开始向上一直requestLayout。最终到达ViewRootImpl。ViewParent 就是当前的传输链。【参见职责链设计模式】

 

第一步。

ViewRootImpl发现请求了布局。那么就会调用measure方法。

measure方法确认当前View是否有FORCE_LAYOUT标记。

如果有,那么就会进行重新measure。并且设置标记LAYOUT_REQUIRED。 

  public final void measure(int widthMeasureSpec, int heightMeasureSpec) {          if ((mPrivateFlags & FORCE_LAYOUT) == FORCE_LAYOUT ||                  widthMeasureSpec != mOldWidthMeasureSpec ||                  heightMeasureSpec != mOldHeightMeasureSpec) {                // first clears the measured dimension flag              mPrivateFlags &= ~MEASURED_DIMENSION_SET;                if (ViewDebug.TRACE_HIERARCHY) {                  ViewDebug.trace(this, ViewDebug.HierarchyTraceType.ON_MEASURE);              }                // measure ourselves, this should set the measured dimension flag back              onMeasure(widthMeasureSpec, heightMeasureSpec);                // flag not set, setMeasuredDimension() was not invoked, we raise              // an exception to warn the developer              if ((mPrivateFlags & MEASURED_DIMENSION_SET) != MEASURED_DIMENSION_SET) {                  throw new IllegalStateException("onMeasure() did not set the"                          + " measured dimension by calling"                          + " setMeasuredDimension()");              }                mPrivateFlags |= LAYOUT_REQUIRED;          }            mOldWidthMeasureSpec = widthMeasureSpec;          mOldHeightMeasureSpec = heightMeasureSpec;      }

 

 

第二步。

 在随后的layout方法中,会判断这个标记。如果这个标记为true。

那么就一定会调用onLayout.

onLayout调用后清理LAYOUT_REQUIRED标记。

layout调用之后,会清理掉FORCE_LAYOUT标记。

  @SuppressWarnings({"unchecked"})     public void layout(int l, int t, int r, int b) {         int oldL = mLeft;         int oldT = mTop;         int oldB = mBottom;         int oldR = mRight;         boolean changed = setFrame(l, t, r, b);         if (changed || (mPrivateFlags & LAYOUT_REQUIRED) == LAYOUT_REQUIRED) {             if (ViewDebug.TRACE_HIERARCHY) {                 ViewDebug.trace(this, ViewDebug.HierarchyTraceType.ON_LAYOUT);             }               onLayout(changed, l, t, r, b);             mPrivateFlags &= ~LAYOUT_REQUIRED;               ListenerInfo li = mListenerInfo;             if (li != null && li.mOnLayoutChangeListeners != null) {                 ArrayList
listenersCopy = (ArrayList
)li.mOnLayoutChangeListeners.clone(); int numListeners = listenersCopy.size(); for (int i = 0; i < numListeners; ++i) { listenersCopy.get(i).onLayoutChange(this, l, t, r, b, oldL, oldT, oldR, oldB); } } } mPrivateFlags &= ~FORCE_LAYOUT; }

 

 

当然在上述过程中,影响到了兄弟或者是父亲View的大小, 那么也兄弟或者是父亲View也会调用layout/onLayout。不管其是否已经调用requestLayout。如果说指定的MeasureSpec为此也发生了变化,

那么measure/onMeasure也会被调用。

 

通过上述分析发现,只要调用了requestlayout, 那么measure和onMeasure,以及layout,onlayout,draw onDraw都会被调用。

 

在很多情况下,requestLayout是不需要被调用的。例如,我们把一个AbsoluteLayout里面的childView挪动一下位置。我们仅仅需要调用的可能就是重新布局当前AbsoluteLayout,然后调用invalidate方法进行重绘。而不是从当前View向上的整个View树形结构都要重新layout,onLayout,measure,onMeasure一次。

这个时候,怎么办?

 

一种方法是,直接调用onLayout。然后调用invalidate进行重绘。很明显可以提升绘制效率。由于父View的layout实现中对会通知布局的listener。但是由于无法得到listener,因此调用onlayout的时候无法对其进行通知,这也是这种实现的缺陷。

 

转载于:https://www.cnblogs.com/huanyou/p/5759718.html

你可能感兴趣的文章
准备学习一下移动网站开发
查看>>
KeyMob移动广告聚合平台服务_广告聚合平台_工具
查看>>
Linux 下 项目管理
查看>>
Steve Souders 高性能网站建设指南14条提升网站页面速度的最佳实践(1)
查看>>
ORA-00932: 数据类型不一致: 应为 NUMBER, 但却获得 BINARY
查看>>
YMP开发框架快速上手(一)
查看>>
Oracle11g RAC下ASM 的管理与维护
查看>>
centos 安装ftp
查看>>
android 简单的 左右上下手势 判断
查看>>
CSS重置, 批量设置指定所有类型控件的CSS风格
查看>>
Go 1.4 src/pkg → src
查看>>
成为JavaGC专家Part I — 深入浅出Java垃圾回收机制
查看>>
linux 配置软连接的需要注意的一个问题
查看>>
如何对system.img和userdata.img解包,再重新打包
查看>>
都是 HBase 上的 SQL 引擎,Kylin 和 Phoenix 有什么不同?
查看>>
使用jMeter构造大量并发HTTP请求进行微服务性能测试
查看>>
实现RabbitMQ 延时消息
查看>>
系统目录结构、ls命令、文件类型、alias命令
查看>>
京东八年架构师: Redis 如何分布式,金融的设计原理
查看>>
完美Python入门基础知识点总结,看完你的Python就完全入门了!
查看>>