androidTestImplementation ‘com.android.support.test:rules:1.0.2’
引用。
上面runClick里面的测试,一个表示text内容为"点击获取城市感冒指数"则执行点击,另一个表示找到R.id.wendu的控件,然后看它是否text内容为"点击获取城市今天温度"。
================================================================
这两个都是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是一款专门针对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的对象需要植入到应用代码中,从而才能执行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
}
}