像 google 一样测试系列之四:技术篇

作者:郑小辉 团队:腾讯移动品质中心TMQ

引言

Android白盒测试覆盖率低的最主要原因,是大部分人都没有测到Android层,只测试了Java层部分,导致覆盖率低。因为大部分代码都在Android层。

亲,你是不是认为Android层的都测试不了尼,下面来看看吧

一 、Android层 可测性预研

1、Application可测性

Application级 是app全局共享的,通常用作数据传递,数据共享,数据缓存等。

因此,application级也是需要测试的,也是可测的。

样例代码如下:

验证版本号等等。

结论:Application级可测。

2、Activity可测性

Activity可测性,主要预研,Activity里的:

1)private方法是否可测(不管被UI层调用的);

2)public 方法是否可测;

3)static 方法是否可测;

4)相关依赖是否可mock;

5)接收参数的Activity是否可测。

(1)private方法是否可测(不管是否被UI层调用的)。

业务代码中,private 方法通常被UI层调用,如下,被按back键时调用,如果按一般的测试java层是测试不到的。

如果你又想通过反射来测试:

要么你要得到当前业务app的该Activity实例,这个一般人是得不到的;

要么又想自己创建一个Activity来反射,但因android机制,即使你能创建出来,那这个Activity实例也不是业务app的该Activity实例。方法依赖其他逻辑照样测试不了。

可测性预研结果:可测。

测试样例代码如下:

先创建Activity的rule,然后通过rule.getActivity()获得实例。再获取该page。

有了对象,再反射调用就可以了,最后验证结果。

结论:Activity里的 private方法可测。

(2)public 方法是否可测。

(3)static 方法是否可测。

得到了实例,public和static均可以直接调用。

结论:Activity里的public 返回和static返回 可测。

(4)相关依赖是否可mock。

如下,purify业务app中,有第三方库的调用,这个调用里存在异步线程的处理。如果不mock,将不能得到正确的验证结果。

mock后的测试样例代码如下:

结论: 可Mock。

(5)接收参数的Activity是否可测。

如下,业务代码中,Activity启动时含有对启动intent是否有参数的逻辑。如含有对应参数,则上报一个统计点。

那么这个逻辑是否可测呢?

测试样例代码如下:

结论:接收参数的Activity可测。

至此,Activity里大部分逻辑都可测。

3、Service可测性

Service进程是app的一个单独进程,里面的不少逻辑,同样很多人人为不可测,那实际尼?

Service测试样例代码如下:

结论:Service可测。

4、广播Broadcast可测性

假设有一业务广播如下:

同样的,广播测试样例代码如下:

结论:broadcast可测。

二、java层单元和接口测试

纯java的逻辑测试,是大部分人做的,但这里所要说的,还包含一些依赖android环境的测试,比如,一个java方法依赖了android 的context,SharedPreferences等,可能有人就又认为这些测不了了。

单元测试:包含类的测试,主要测试多条件入参的测试,比如一个类方法 不同参数的传入测试。

接口测试:包含调用链路的测试,包括不同层次的链路调用。主要测试集成路径,不同参数的路径。

单元测试和接口测试,大部分做过白盒的都懂,这里就不细说。

主要说说 ,涉及Android部分的如何测试。

如下业务代码:

被测方法isNeedUpdateTrashRules -->调用了 retrieveTrashRulesUpdateMillis,而该方法用了Android环境SharedPreferences。

思路还是:mock掉,然后塞进去,最后验证。

测试样例代码如下:

三、异步线程可测性

被测方法调用了异步代码时,测试代码将无法正确的验证结果。导致用例失败或不可测。

因此,如何能让异步代码可测,也是如何让现有代码更可测的一部分。

异步线程的可测性思路。

思路一:通过CountDownLatch来实现,这个需要改业务代码,一般不怎么用。

思路二:mock掉异步对象,反射进去,当执行异步时,通过调用拦截获得thread对象,立即调用thread.run()。

思路三:new thread的方式,一般都和回调一起,先mock掉父调用,拦截回调,直接调用回调。

下面说下使用较多的 ExecuteService异步和handle.post的测试样例:ExecuteService异步的测试样例。

业务有如下图异步线程:

测试样例如下:

handle.post() 样例:

如下,业务代码使用了内部handle来处理消息,当执行到handle.post() 因为是异步,测试用例无法获取正常结果。

然而 handle.post有如下多个方法,难道每个都要针对性处理么?那这样比较麻烦。

翻看Android 源码handle.post部分,发现如下调用关系:

各方法最终都调用了 handle.sendMessageAtTime(Message,long),因此只要mock这个就可以了。

测试样例代码如下:

四、函数回调可测性

思路:依然是通过mock,并拦截函数调用,获取对象直接调用。

1、参数传入回调方式可测性

如下业务代码:原始回调被包装了3次回调,最后以参数方式传入。

测试样例代码:

五、Mock简单举例

Android 白盒测试mock,支持多种框架,常用的用mockito和PowerMock。

其中静态方法的mock只能用PowerMock。

Local Unit Tests :可用mockito和PowerMock。

Instrumented Tests :只能用mockito。

1、Android环境Mock

Android Context的mock:

2、Android API Mock

3、普通Mehod和Field Mock

普通对象的method调用:

普通Field的mock:

4、静态Method和Field Mock

静态要mock的,需用PowerMock。网上很多资料。这里就不细说了。

六、反射调用与执行

但遇到 private 标示的逻辑时,不能直接调用。但又想把他测试覆盖,可用反射来执行。

一般主要在:

1、不好从上层介入的地方:private 标示的逻辑,被UI层调用,比如被onclick事件调用,你不好从上层介入。

2、多参数或分支较多:private标示的逻辑,入参较复杂,内部分支和逻辑较多,想以单测函数来先保证正确性。

业务代码的反射和 lib库代码的反射,用法差不多。就写在一起了: ReflectUtil是已经封装好的反射工具类。

七、业务代码直接调用

在模式和方案选型时,是否能直接调用业务代码,也是一个衡量项。最好是能直接调用。能省事省力。

1、业务代码直接调用

未完待续......

搜索微信公众号:腾讯移动品质中心TMQ,获取更多测试干货!

原创声明,本文系作者授权云+社区-专栏发表,未经许可,不得转载。

如有侵权,请联系 yunjia_community@tencent.com 删除。

编辑于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Golang语言社区

Go语言是彻底的面向组合的并发语言

面向组合编程从AOP的Mixin,然后到Ruby的Traits,直至DCI设计,包括Scala的trait的组合设计,这些都有一个共同特点,组合特性是显式的,也...

3126
来自专栏腾讯Bugly的专栏

那些年,我们一起写过的“单例模式”

本文来自:“天天P图攻城狮”公众号(ttpic_dev) 题记 度娘上对设计模式(Design pattern)的定义是:“一套被反复使用、多数人知晓的、经过分...

3874
来自专栏求索之路

Android源码设计模式解析与实战笔记

1.单一职责原则:比如说一个ImageLoader,需要加载图片的缓存图片,此时如果将这两个功能都放在一个类中,就违反了这个原则, 我们需要将不同的功能用类精...

3365
来自专栏撸码那些事

最佳编码实践——单一职责原则

同时应用这些最佳实践,可以提升代码适应变更的能力。但是凡事要有度,过度使用虽然可以让代码有很高的自适应能力,但是会导致层次粒度过小而难以理解或使用,还会影响代码...

896
来自专栏恰同学骚年

设计模式的征途—7.适配器(Adapter)模式

在现实生活中,我们的笔记本电脑的工作电压大多数都是20V,而我国的家庭用电是220V,如何让20V的笔记本电脑能够工作在220V的电压下工作?答案:引入一个电源...

833
来自专栏JAVA烂猪皮

BAT面试常的问题和最佳答案

客户端发出http请求,web服务器将请求转发到servlet容器,servlet容器解析url并根据web.xml找到相对应的servlet,并将reques...

762
来自专栏小曾

.Net 如何模拟会话级别的信号量,对http接口调用频率进行限制(有demo)

现在,因为种种因素,你必须对一个请求或者方法进行频率上的访问限制。 比如, 你对外提供了一个API接口,注册用户每秒钟最多可以调用100次,非注册用户每秒钟最...

772
来自专栏架构之路

追源索骥:透过源码看懂Flink核心框架的执行流程

写在最前:因为这篇博客太长,所以我把它转成了带书签的pdf格式,看起来更方便一点。想要的童鞋可以到我的公众号“老白讲互联网”后台留言flink即可获取。

2433
来自专栏xingoo, 一个梦想做发明家的程序员

选择模型2

选择模型 fd_set结构可以把多个套接字连在一起,形成一个套接字集合 typedef struct fd_set{ u_int fd_count;//下面数组...

18010
来自专栏编程札记

深入golang之---goroutine并发控制与通信

本文章通过goroutine同步与通信的一个典型场景-通知子goroutine退出运行,来深入讲解下golang的控制并发。

4536

扫码关注云+社区