Android单元测试利器–Robolectric 结合powermock测试

PowerMock测试Demo

前面的章节中有提到过Robolectric 3.0不能直接针对非Android Sdk的类做Shadow,必须使用PowerMock或者mockito处理,PowerMock支持静态函数的mock,还支持partialmock,也就是说mock某个类时,不需要为这个类的所有函数做mock处理,只需针对需要改变行为的函数进行mock就可以了,其它函数执行时还是mock之前的逻辑。这点非常有用,否则每次使用powermock或者mockito时需要针对某个类的所有函数都要处理,如果函数比较多,那会相当麻烦。

Android单元测试系列文章的代码都可以在Github上找到:https://github.com/cloudchou/RobolectricDemo

Robolectric 3.1已支持针对非AndroidSdk的类做Shadow,但是不支持Powermock。因为创建Shadow类的方式,需要写的代码比PowerMock方式多很多,所以我们建议使用PowerMock+Robolectric3.0+mockito做单元测试。

接下来我们看如何使用PowerMock做Partial Mock。

首先看一下要被mock的类的代码:

//被测试类只是简单返回一些字符串

public class HelloUtils2 {

????public static String test1() { return "Hello Utils2 test1"; }

????public static String test2() { return "Hello Utils2 test2"; }

}

测试代码如下所示:

?//不可缺少的代码 表明使用Powermock执行单元测试,虽然我们使用的是RoblectricGradleTestRunner来执行单元测试

//但是添加了如下代码后RoblectricGradleTestRunner会调用PowerMock的TestRunner去执行单元测试

@RunWith(RobolectricGradleTestRunner.class)

@Config(constants = BuildConfig.class, sdk = 21)

//必须写如下代码 让PowerMock 忽略Robolectric的所有注入

@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})

//因为我们是针对类做静态函数的mock,所以必须使用PrepareForTest说明我们要mock的类

@PrepareForTest({HelloUtils2.class})

public class PartialPowerMockTest {

????@Rule

????public PowerMockRule rule = new PowerMockRule();?


? ??//因为我们是针对类做静态函数的mock,所以必须在所有测试用例执行之前? ?

????@Before

????public void setup() {

?????????//使用PowerMockito.mockStatic开启对HelloUtils2的静态mock

????????PowerMockito.mockStatic(HelloUtils2.class);

????}

????@Test

????public void testPartialmock() throws Exception {

????????//调用spy表明是partial mock

????????PowerMockito.spy(HelloUtils2.class);

????????//当执行HelloUtils2.test1函数时,让它返回it's partial mocked

????????PowerMockito.doReturn("it's partial mocked").when(HelloUtils2.class, "test1");

????????//test1 函数的行为已改变 会返回 it's partial mocked

????????System.out.println(HelloUtils2.test1());

????????System.out.println(HelloUtils2.test2());

????}

}

执行代结果如下所示:

PowerMock还可以用于做校验,比如测试某个函数时,校验该函数调用的其它函数是否执行了指定的次数,或者校验这些函数是否执行超时。

Powermock进行校验时,和mockito做校验有比较大的区别,需要先执行测试的函数的逻辑, Powermock会收集执行时的数据,比如函数被调用多次,函数执行时间等信息,然后再对Powermock收集到的数据进行校验 , verifyStatic函数的参数是一个校验模型。times(3)表示执行了3次, 但是此时还不知道对哪个函数的执行次数校验3次,必须在后面调用 要校验的 函数, 执行后, Powermock就知道要校验谁了,Powermock此时会执行真正的校验逻辑。

示例代码如下所示:

//不可缺少的代码 表明使用Powermock执行单元测试,虽然我们使用的是RoblectricGradleTestRunner来执行单元测试

//但是添加了如下代码后RoblectricGradleTestRunner会调用PowerMock的TestRunner去执行单元测试

@RunWith(RobolectricGradleTestRunner.class)

@Config(constants = BuildConfig.class, sdk = 21)

//必须写如下代码 让PowerMock 忽略Robolectric的所有注入

@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})

//因为我们是针对类做静态函数的mock,所以必须使用PrepareForTest说明我们要mock的类

@PrepareForTest({HelloUtils2.class, SLog.class})

public class PowerMockVerifyTest {?

????@Rule

????public PowerMockRule rule = new PowerMockRule();


????//因为我们是针对类做静态函数的mock,所以必须在所有测试用例执行之前?

????@Before

????public void setup() {

?????//使用PowerMockito.mockStatic开启对HelloUtils2的静态mock

????????PowerMockito.mockStatic(HelloUtils2.class);

????????PowerMockito.mockStatic(SLog.class);

????}

????@Test

????public void testVerify() throws Exception {

????//先执行逻辑, Powermock会收集执行时的数据,比如函数被调用多次,函数执行时间等信息, HelloUtils2.test3(); HelloUtils2.test3(); HelloUtils2.test3();

????//然后再对Powermock收集到的数据进行校验 , verifyStatic函数的参数是一个校验模型

????// times(3)表示执行了3次, 但是此时还不知道对哪个函数的执行次数校验3次

????// 必须在后面调用 要校验的 函数, 执行后, Powermock就知道要校验谁了,

????//Powermock此时会执行真正的校验逻辑, 看test3函数是否真的执行了3次

????????PowerMockito.verifyStatic(times(3));

????????HelloUtils2.test3();

????}

????@Test

????public void testVerifyFailed() throws Exception {

????????//先执行逻辑, Powermock会收集执行时的数据,比如函数被调用多次,函数执行时间等信息,

????????HelloUtils2.test3();

????????HelloUtils2.test3();

????????// HelloUtils2.test3();

????????//然后再对Powermock收集到的数据进行校验 , verifyStatic函数的参数是一个校验模型

????????// times(3)表示执行了3次, 但是此时还不知道对哪个函数的执行次数校验3次

????????// 必须在后面调用 要校验的 函数, 执行后, Powermock就知道要校验谁了,

????????//Powermock此时会执行真正的校验逻辑, 看test3函数是否真的执行了3次

????????PowerMockito.verifyStatic(times(3));

????????HelloUtils2.test3();

????}

}

执行testVerifyFailed的结果如下所示:


我们不仅可以对函数调用次数进行校验, 还可以对函数的参数做限制,也就是说校验指定参数的函数校验调用次数,示例代码如下所示:

@RunWith(RobolectricGradleTestRunner.class)

@Config(constants = BuildConfig.class, sdk = 21)

@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})?

@PrepareForTest({HelloUtils2.class, SLog.class})

public class PowerMockVerifyTest {

????@Test

????public void testVerifyStringParam() throws Exception {

????????HelloUtils2.testStringParam("test"); // 我们如果将下面对testStringParam函数调用的参数设置为其它值,则该测试用例会执行失败

????????HelloUtils2.testStringParam("test");

????????PowerMockito.verifyStatic(times(2));

????????HelloUtils2.testStringParam("test");

????}?

}

有了这个校验机制, 我们也就能针对Android平台的很多业务场景进行校验了,因为我们通?;嵩诖肜锎蛴og,如果遇到异常场景,我们会打error日志,我们只需要校验是否打印了error日志就可以知道我们的业务逻辑是否符合预期。

但是我们不能直接针对Log进行校验,Robolectric框架对Android sdk的类都做了替换,都设置了Shadow类,比如Log类对应的Shadow类是ShadowLog。所以我们需要对Log进行一层封装,不过在我们的业务逻辑里通常也会针对Log做一层封装,在我们的示例代码里使用了SLog对Log类进行了封装。

先看一下被测试的类的业务逻辑:

public class HelloUtils2 {

????private static final String TAG = HelloUtils2.class.getSimpleName(); //...

????public static void testLog() { SLog.e(TAG, "Hello world"); } //...

}

我们在测试代码里,只需校验调用了SLog,传了指定的参数即可,测试代码如下所示:

@RunWith(RobolectricGradleTestRunner.class)

@Config(constants = BuildConfig.class, sdk = 21)

@PowerMockIgnore({"org.mockito.*", "org.robolectric.*", "android.*"})?

@PrepareForTest({HelloUtils2.class, SLog.class})

public class PowerMockVerifyTest {

????@Rule public PowerMockRule rule = new PowerMockRule();

????@Before

????public void setup() {

????????PowerMockito.mockStatic(HelloUtils2.class);

????????PowerMockito.mockStatic(SLog.class);

????}?

????@Test

????public void testVerifyLog() throws Exception { //针对HelloUtils2类和SLog类做partial mock PowerMockito.spy(HelloUtils2.class);

????????PowerMockito.spy(SLog.class);

????????HelloUtils2.testLog(); //因为testLog函数里调用了SLog, 而我们接下来校验时,不允许对SLog调用,所以校验会显示错误 //使用这种方式能很方便地校验业务逻辑 PowerMockito.verifyStatic(times(0));

????????SLog.e(HelloUtils2.class.getSimpleName(), "Hello world");

????}

}

执行结果如下所示:


?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容