一个 forceLayout() 和 requestLayout() 的测试

两个view:

一个是系统默认的FrameLayout,  A

一个是自己自定义的MyView extends View,重载了onMeasure函数(): B

[html] view plain copy

print?

  1. @Override
  2. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  3. // TODO Auto-generated method stub
  4. super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  5. // Log.e(TAG, "onMeasure " + MeasureSpec.getSize(widthMeasureSpec)
  6. //      + " " + MeasureSpec.getSize(heightMeasureSpec));
  7. setMeasuredDimension(MeasureSpec.makeMeasureSpec(mW-=10, MeasureSpec.EXACTLY),
  8. MeasureSpec.makeMeasureSpec(mH-=10, MeasureSpec.EXACTLY));
  9. }
	@Override
	protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
		// TODO Auto-generated method stub
		super.onMeasure(widthMeasureSpec, heightMeasureSpec);
		// Log.e(TAG, "onMeasure " + MeasureSpec.getSize(widthMeasureSpec)
		// 		+ " " + MeasureSpec.getSize(heightMeasureSpec));
		setMeasuredDimension(MeasureSpec.makeMeasureSpec(mW-=10, MeasureSpec.EXACTLY),
				MeasureSpec.makeMeasureSpec(mH-=10, MeasureSpec.EXACTLY));
	}

逻辑很简单,每次只要B的onMeasure被触发,那么B的尺寸就原来小10.

A包含B,

几个测试case: (measure() 是final的,无法覆盖,不过B的measure()应该是每次被调到的)

1. A.requestLayout();                                    不会调用到B.onMeasure() 和 B.onLayout(), 会调用B.layout() (注意连draw()都没有被调到,这意味着B被A认为完全没变,不需要重绘)

2. B.forceLayout(); A.requestLayout()           会调用到B的.onMeasure() 和 onLayout() 以及 layout(), draw(), onDraw()

3. B.requestLayout()                                      会调用到B的.onMeasure() 和 onLayout() 以及 layout(),draw(), onDraw()

4. B.invalidate()                                               B的layout(), onLayout() 和 onMeasure() 没有被调到,只有 draw() 和 onDraw() 被调到.

5  A.invalidate()                                               B的所有函数都不会被调到.

这个结果也符合code的逻辑,forceLayout()如果注释说的一样,是在下一次layout bypass 的过程(自己不会发起一次layout bypass)中,会强制的重新onMeasure和onLayout()

而requestLayout() 除了干forceLayout()的事情将自己的FORCE_LAYOUT标志位设上外,还会发起一次layout pass,

在layout bypass 从 A 传到 B时,虽然 A 的 onMeasure() 和 onLayout() 会调用 B 的 measure() 和 layout(), 但是,因为 B的layout状态没有什么改变,因此

onMeasure()和 onLayout()不会被调用.

在当前4.4的code:

[html] view plain copy

print?

  1. public void forceLayout() {
  2. ................................................
  3. mPrivateFlags |= PFLAG_FORCE_LAYOUT;
  4. mPrivateFlags |= PFLAG_INVALIDATED;
  5. }
  6. public final void measure(int widthMeasureSpec, int heightMeasureSpec) {
  7. ...............................................................
  8. if ((mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ||
  9. .......................................................
  10. int cacheIndex = (mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ? -1 :
  11. mMeasureCache.indexOfKey(key);
  12. if (cacheIndex < 0 || sIgnoreMeasureCache) {
  13. ..................................................
  14. onMeasure(widthMeasureSpec, heightMeasureSpec);
  15. mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
  16. } else {
  17. long value = mMeasureCache.valueAt(cacheIndex);
  18. // Casting a long to int drops the high 32 bits, no mask needed
  19. setMeasuredDimension((int) (value >> 32), (int) value);
  20. mPrivateFlags3 |= PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
  21. }
  22. ......................................................................................
  23. mPrivateFlags |= PFLAG_LAYOUT_REQUIRED;
  24. }
  25. .............................................................................
  26. }
  27. public void layout(int l, int t, int r, int b) {
  28. .....................................................
  29. if (changed || (mPrivateFlags & PFLAG_LAYOUT_REQUIRED) == PFLAG_LAYOUT_REQUIRED) {
  30. onLayout(changed, l, t, r, b);
  31. mPrivateFlags &= ~PFLAG_LAYOUT_REQUIRED;
  32. ...................................................................
  33. }
  34. mPrivateFlags &= ~PFLAG_FORCE_LAYOUT;
  35. mPrivateFlags3 |= PFLAG3_IS_LAID_OUT;
  36. }
public void forceLayout() {
     ................................................
        mPrivateFlags |= PFLAG_FORCE_LAYOUT;
        mPrivateFlags |= PFLAG_INVALIDATED;
    }

public final void measure(int widthMeasureSpec, int heightMeasureSpec) {
       ...............................................................
        if ((mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ||
               .......................................................
            int cacheIndex = (mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ? -1 :
                    mMeasureCache.indexOfKey(key);
            if (cacheIndex < 0 || sIgnoreMeasureCache) {
                ..................................................
                onMeasure(widthMeasureSpec, heightMeasureSpec);
                mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
            } else {
                long value = mMeasureCache.valueAt(cacheIndex);
                // Casting a long to int drops the high 32 bits, no mask needed
                setMeasuredDimension((int) (value >> 32), (int) value);
                mPrivateFlags3 |= PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
            }
......................................................................................
            mPrivateFlags |= PFLAG_LAYOUT_REQUIRED;
        }
.............................................................................
}

public void layout(int l, int t, int r, int b) {
        .....................................................
        if (changed || (mPrivateFlags & PFLAG_LAYOUT_REQUIRED) == PFLAG_LAYOUT_REQUIRED) {
            onLayout(changed, l, t, r, b);
            mPrivateFlags &= ~PFLAG_LAYOUT_REQUIRED;

           ...................................................................
        }

        mPrivateFlags &= ~PFLAG_FORCE_LAYOUT;
        mPrivateFlags3 |= PFLAG3_IS_LAID_OUT;
    }

可见forceLayout()可以导致onMeasure() 和 onLayout(). 而requestLayout() 干的事情比forceLayout()只多不少.

View的forceLayout后面一般会紧跟着View的measure(), 这样可以把view的measuredSize 通过 setMeasuredDimension 设上.

而PFLAG_INVALIDATED 这个flag 应该是标记 重绘的。

而requestLayout()/forceLayout() 设置的另一个flag PFLAG_FORCE_LAYOUT,也一定可以在measure()中 将PFLAG_LAYOUT_REQUIRED flag给打上,

而PFLAG_LAYOUT_REQUIRED 则是可以保证了在调用 layout()函数时,onLayout() 函数会被调用到.

还有一点,之前没有仔细看, View的measure(A, B) 函数:

[html] view plain copy

print?

  1. 如果有PFLAG_FORCE_LAYOUT 或者 本次的measure的尺寸 A, B 和 之前的尺寸不一样
  2. if ((mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ||
  3. widthMeasureSpec != mOldWidthMeasureSpec ||
  4. heightMeasureSpec != mOldHeightMeasureSpec) {
  5. ........................................................
  6. 这一步其实是从之前的MeasureCache里找是否存在 和 本次的measure尺寸一致的 cache
  7. int cacheIndex = (mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ? -1 :
  8. mMeasureCache.indexOfKey(key);
  9. 如果没有找到,那么就调用onMeasure, 一般来说,调用了onMeasure才能真正的setMeasuredDimension
  10. if (cacheIndex < 0 || sIgnoreMeasureCache) {
  11. // measure ourselves, this should set the measured dimension flag back
  12. onMeasure(widthMeasureSpec, heightMeasureSpec);
  13. mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
  14. } else {
  15. 如果找到有cache,那么就将cache的值 设为 setMeasuredDimension
  16. long value = mMeasureCache.valueAt(cacheIndex);
  17. // Casting a long to int drops the high 32 bits, no mask needed
  18. setMeasuredDimension((int) (value >> 32), (int) value);
  19. mPrivateFlags3 |= PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
  20. }
  21. }
  22. ..............................................
  23. mPrivateFlags |= PFLAG_LAYOUT_REQUIRED;
  24. ..............................................
  25. mOldWidthMeasureSpec = widthMeasureSpec;
  26. mOldHeightMeasureSpec = heightMeasureSpec;
  27. mMeasureCache.put(key, ((long) mMeasuredWidth) << 32 |
  28. (long) mMeasuredHeight & 0xffffffffL); // suppress sign extension
	       如果有PFLAG_FORCE_LAYOUT 或者 本次的measure的尺寸 A, B 和 之前的尺寸不一样
               if ((mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ||
				widthMeasureSpec != mOldWidthMeasureSpec ||
				heightMeasureSpec != mOldHeightMeasureSpec) {

			........................................................

			这一步其实是从之前的MeasureCache里找是否存在 和 本次的measure尺寸一致的 cache
                        int cacheIndex = (mPrivateFlags & PFLAG_FORCE_LAYOUT) == PFLAG_FORCE_LAYOUT ? -1 :
					mMeasureCache.indexOfKey(key);
			如果没有找到,那么就调用onMeasure, 一般来说,调用了onMeasure才能真正的setMeasuredDimension
                        if (cacheIndex < 0 || sIgnoreMeasureCache) {
				// measure ourselves, this should set the measured dimension flag back
				onMeasure(widthMeasureSpec, heightMeasureSpec);
                                mPrivateFlags3 &= ~PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
                        } else {
                               如果找到有cache,那么就将cache的值 设为 setMeasuredDimension
                               long value = mMeasureCache.valueAt(cacheIndex);
                               // Casting a long to int drops the high 32 bits, no mask needed
                               setMeasuredDimension((int) (value >> 32), (int) value);
                               mPrivateFlags3 |= PFLAG3_MEASURE_NEEDED_BEFORE_LAYOUT;
                        }
                 }

                ..............................................
                mPrivateFlags |= PFLAG_LAYOUT_REQUIRED;

                ..............................................
                mOldWidthMeasureSpec = widthMeasureSpec;
                mOldHeightMeasureSpec = heightMeasureSpec;

                mMeasureCache.put(key, ((long) mMeasuredWidth) << 32 |
                      (long) mMeasuredHeight & 0xffffffffL); // suppress sign extension 

这说明调用measure(A, B), 只要尺寸变化,都会影响 MeasuredHeight/Width

借鉴:http://blog.csdn.net/fyfcauc/article/details/41895509

时间: 2024-10-03 18:16:25

一个 forceLayout() 和 requestLayout() 的测试的相关文章

Selenium也是一个用于Web应用程序测试的工具

Selenium也是一个用于Web应用程序测试的工具.Selenium测试直接运行在浏览器中,就像真正的用户在操作一样.支持的浏览器包括IE.Mozilla Firefox.Mozilla Suite等.这个工具的主要功能包括:测试与浏览器的兼容性--测试你的应用程序看是否能够很好得工作在不同浏览器和操作系统之上.测试系统功能--创建衰退测试检验软件功能和用户需求.支持自动录制动作和自动生成.Net.Java.Perl等不同语言的测试脚本.Selenium 是ThoughtWorks专门为Web

火狐上的一个post提交工具(主要用于测试接口时候)

添加的过程 安装完后,就可以在下图上,看到一个poster 点击poster就可以看到下图 图中红线圈好的,是必须要填写的 Url是访问路径 Name是参数名称 Value是参数值 需要注意一点的是:如图 切换上面的Tab控件需要点击图中红线圈住的部分,这样才能使的参数添加到url后面传参,丢掉这一部分的话,参数是没有办法传过去的! 火狐上的一个post提交工具(主要用于测试接口时候),布布扣,bubuko.com

一个性能(压力)测试工具列表

我个人认为一个优秀的软件工程师必须掌握一定的测试技能.可能有的开发工程师认为测试是测试工程师的事情,不是开发工程师的事情.这种开发工程师是缺一条腿的,离开了测试工程师,你如何证明你的程序没有问题. 下面是我这两天搜集的性能测试工具,如果大家知道别的工具,可以在回复中分享出来. Webbench Web Bench is very simple tool for benchmarking WWW or proxy servers. Uses fork() for simulating multip

一个好用的压力测试工具tsung

一个好用的压力测试工具tsung 前段时间一直在忙各种事情,快三周没弄过引擎了,今天有点时间,正好之前写的服务器引擎也到了收尾测试的阶段,于是就研究了下怎么测试服务器压力.        目前主流用来测试服务器压力的工具有webbench,ab等,但是没法满足我的需求,webbench我没找到post json的方法,ab主要是用来测试apache的.经过向一些群里的朋友询问,他们推荐了erlang编写的tsung.百度了一下这个东东,发现百度百科是这么描述的:看起来挺不错,下载研究下.    

从一个实例详解敏捷测试的最佳实践

简介: 敏捷软件开发是目前十分流行,并在业界逐步推广的软件开发模式.不同与传统的软件开发模式,敏捷开发模式有着自己鲜明的价值和方法.其中,敏捷测试部分也同以往的软件测试流程有所不同.这对测试人员提出了新的要求,带来了新的挑战.本文将结合一个软件项目实例,基于项目开发的不同阶段,详细介绍每个阶段的主要测试活动.文中将分析每个主要测试活动的前提条件和目标任务,并根据实例推荐最佳的解决方案. 第一部分:敏捷软件开发简介 敏捷软件开发(Agile Software Development)初起于九十年代

一个&quot;如何使用示波器安全测试接市电电路板&quot;的问题

最近犯了一个错误测试操作: 测试场景:直接从市电插座取电接入3W非隔离开关电源电路板,使用示波器测试输出电压,此时示波器通过另外一个插座直接从市电取电 测试后果:在将示波器接到输出负极的一瞬间,漏电保护开关跳闸了!!! 经过咨询.分析,才明白自己由于没搞懂“示波器地线与市电的零线.火线还有大地址间的关系”才导致错误的测试操作,该操作不仅会让市电跳闸,还可能会把示波器烧坏! 如下图截取广州致远电子股份有限公司<如何用示波器安全测量市电>,该图表现出火线.零线和地线的关系: 火线(L):也称相线,

一个简单的binlog恢复测试

日常的数据备份及恢复测试,是DBA工作重中之重的事情,所以要做好备份及测试,日常的备份常见有mysqldump+binlog备份.xtrabackup+binlog备份,无论那一种,几乎都少不了对binlog的备份,说明了binlog在数据恢复中的重要性,下面做个小测试,是工作中不少运维或者新人DBA容易犯的错. 创建一个测试表tb1: <test>([email protected]) [xuanzhi]> show create table tb1\G ***************

一个简单的函数指针测试例子

一般函数指针的一个简单测试.将函数放到vector里面,将函数作为形参. 1 typedef void(*GTestFunPtr)(int, int); 2 3 void test1(int a, int b) 4 { 5 std::cout << "test1:" << a + b << std::endl; 6 } 7 8 void test2(int a, int b) 9 { 10 std::cout << "test2

[python测试框架学习篇] 分享一个和adb相关的测试框架

https://testerhome.com/topics/7106   (user: zteandallwinner     password: same to qq ) 264768502 · #1 · 2017年01月13日 小小的建议 没缩进不能看,不如贴gist 单纯的adb的封装有很多人写了 比如我(#厚脸皮) https://github.com/264768502/adb_wrapper 比如这贴: https://testerhome.com/topics/6938 如果要处理