Android测试:关于junit、espresso、mockito、robolectric

androidTestImplementation ‘com.android.support.test:rules:1.0.2’

引用。

上面runClick里面的测试,一个表示text内容为"点击获取城市感冒指数"则执行点击,另一个表示找到R.id.wendu的控件,然后看它是否text内容为"点击获取城市今天温度"。

mockito

================================================================

这两个都是Android测试框架,主要是可以脱离Android设备调试环境,在纯Java环境下面完成Android测试用例。也就是说不像espresso那样需要连接一台android设备了,我们可以像运行JUnit那样来运行android测试用例

在junit单元测试中,需要手动构造测试中对对象的依赖。如A对象方法依赖B对象方法,在测试A对象的时候,我们需要首先构造出B对象,这样子增加了测试的难度,如果依赖过多,相应地也增大了编写测试用例的难度。

Mockito是一个Java开源的测试框架,Mockito在测试中尝试移除我们传统JUnit单元测试中使用的Expect方式,这样子有效降低代码的耦合。使得我们只测试我们需要的方法和类,而不需过多的考虑依赖类

Mockito不能模拟final类、匿名类和Java基本类型;对于final方法和static方法,不能对其 when(…).thenReturn(…) 操作。

另外mock对象,大多都需要植入到应用代码中,从而进行verify(…)操作;但应用代码中不一定有相应的set方法,如果要植入,就需要为了测试添加应用代码。也就是说A对象中需要增加一个set方法专门用来引入mock过的B对象,这样也就是说需要为一个测试用例在原来的对象中增加set方法,这样也是有点复杂。后面我们举例可以用反射来解决这个问题。

@Test

public void testList() {

List mockedList = mock(List.class);

mockedList.add(“one”);

mockedList.clear();

verify(mockedList).add(“one”);

verify(mockedList).clear();

}

适用范围:

解决依赖问题,一般用来测试费生命周期相关的类,比如MVP的P层和M层。如果涉及到生命周期相关的,这需要用到下面的Robolectric测试框架。

Robolectric

====================================================================

Robolectric是一款专门针对Android SDK的测试框架,我们只需要在Java环境中即可运行。

public class WeatherActivityTest {

private WeatherActivity activity;

private WeatherPresenter presenter;

@Before

public void setUp() {

activity = Robolectric.setupActivity(WeatherActivity.class);

}

@Test

public void testToSettingPage(){

activity.toSettingPage();

Intent expectedIntent = new Intent(activity, SettingActivity.class);

Intent actualIntent = ShadowApplication.getInstance().getNextStartedActivity();

Assert.assertEquals(expectedIntent.getComponent(), actualIntent.getComponent());

}

}

通过Robolectric.setupActivity启动一个activity,当然这个activity不会真正的启动。

通过testToSettingPage方法,我们测试activity的toSettingPage方法启动后,是否真的去了SettingActivity这个页面。

适用范围:

有android生命周期的,当然Robolectric功能点不知这些,有兴趣的可以去官网看看:http://robolectric.org/

综合应用

=============================================================

以上四个测试框架各有优点,我们实际应用的时候往往是几种框架一并使用。

比如android开发过程中,我们为了避免使用到android设备来测试,所以我们会用到Robolectric框架,同时也会用到JUnit和Mockito,下面我们看下具体的例子:

@RunWith(RobolectricTestRunner.class)

@Config(sdk = 28)

public class WeatherActivityTest {

private WeatherActivity activity;

private WeatherPresenter presenter;

@Before

public void setUp() {

activity = Robolectric.setupActivity(WeatherActivity.class);

presenter = mock(WeatherPresenter.class);

activity.setPresenter(presenter);//如果不用这种方式就要用反射获取private成员变量

}

@Test

public void testGanmaoClick() {

Button ganmao = activity.findViewById(R.id.ganmao);

ganmao.performClick();

verify(presenter).getGanmao();

//如果不用activity.setPresenter(presenter)则使用下面代码

// try {

// Field field = WeatherActivity.class.getSuperclass().getDeclaredField(“mPresenter”);

// field.setAccessible(true);

// field.set(activity, presenter);

// ganmao.performClick();

// verify(presenter).showTest2();

// } catch (Exception e) {

// //error

// }

// verify(presenter).showTest1();

}

@Test

public void testWenduClick() {

Button wendu = activity.findViewById(R.id.wendu);

wendu.performClick();

assertThat(wendu.getText().toString(), is(“点击获取城市今天温度”));

assertEquals(“验证温度”, wendu.getText().toString(), “点击获取城市今天温度”);

}

@Test

public void testShadows() {

TextView wendu = activity.findViewById(R.id.wendu);

ShadowTextView stv = Shadows.shadowOf(wendu);

assertEquals(“验证温度示”,stv.innerText(),“点击获取城市今天温度”);

}

@Test

public void testToSettingPage(){

activity.toSettingPage();

Intent expectedIntent = new Intent(activity, SettingActivity.class);

Intent actualIntent = ShadowApplication.getInstance().getNextStartedActivity();

Assert.assertEquals(expectedIntent.getComponent(), actualIntent.getComponent());

}

@Test

public void testGetId() {

when(presenter.getId()).thenReturn(100);

}

@Test

public void testList() {

List mockedList = mock(List.class);

mockedList.add(“one”);

mockedList.clear();

verify(mockedList).add(“one”);

verify(mockedList).clear();

}

@Test

public void testOnDestroy() {

activity.initDestroy();

verify(presenter).detachView();

}

}

一些解决方案(mock植入和反射)

==========================================================================

前面说到的mock的对象需要植入到应用代码中,从而才能执行verify操作。但是往往植入的应用代码没有入口,所以我们要手动写一个setXXX,但是为了测试用例专门写一个这个方法也是比较不划算的。mock对象在应用代码里面往往是private方式声明的成员变量,因此我们可以使用反射的方式,将mock对象植入应用代码中。看实例:

@RunWith(RobolectricTestRunner.class)

@Config(sdk = 28)

public class SettingActivityTest {

private SettingActivity activity;

private SettingController controller;

@Before

public void setUp() {

activity = Robolectric.setupActivity(SettingActivity.class);

controller = mock(SettingController.class);

}

@Test

public void testSettingClick() {

Button toasting = activity.findViewById(R.id.toasting);

toasting.performClick();

try {//使用反射获取private成员变量, 否则在SettingActivity就得有个入口将SettingController实例传入

Field field = WeatherActivity.class.getSuperclass().getDeclaredField(“controller”);

field.setAccessible(true);

field.set(activity, controller);

toasting.performClick();

verify(controller).getTitle();

} catch (Exception e) {

//error

}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值