前言
上一篇文章,我们了解了Android测试app的一些框架,这一篇我们来构建出我们的单元测试。
构建本地单元测试
添加依赖
build.gradle的依赖中添加junit4框架和mockito框架
dependencies {
// Required -- JUnit 4 framework
testCompile 'junit:junit:4.12'
// Optional -- Mockito framework
testCompile 'org.mockito:mockito-core:1.10.19'
}
JUnit不必多说是单元测试框架,mockito是java最流行的mock框架。由于本地的单元测试环境中并没有实际运行的Android的环境,所以对于一些单测用例我们是需要这个环境的,但是用真机测试效率又太低,因此引入mock来模拟这个环境。
JUnit的使用
JUnit4比JUnit3更简洁,不需要集成TestCase类,方法名也不需加“test”前缀。我们是@Test注解来标记要测试的方法。
class ExampleUnitTest {
@Test
@Throws(Exception::class)
fun addition_isCorrect() {
assertEquals(4, AppUtils().addNumber(2, 2))
}
}
class AppUtils{
fun addNumber(a : Int, b : Int) = a + b
}
我们可以写多个用例来测试我们的代码。JUnit还有其他的注解来满足我们的测试用例编写。具体的使用可以看这里JUni最新api。这里不再介绍。有机会再写一篇关于JUnit实战的一些技巧。
Not Mocked
JUnit的android插件下运行的是android.jar实际上不包含任何实际的代码,因此是不能直接使用的。如果直接使用,例如是用TestUtils的方法。结果会抛出not mocked的错误。
@Test
fun androidCodeTest() {
assert(TextUtils.isEmpty(""))
}
对于这类的方法,最好的方案是重新写一个自己的TextUtils,使用自己的方法。或者测试用例不使用和android相关的代码。当然也是有绕过这个错误的。
android{
...
testOptions {
unitTests.returnDefaultValues = true
}
}
不过最好不要这样做,这样会导致所有的这种类型都会返回默认值。TextUtils.isEmpty("haha")返回的结果也是true。
使用mockito
1、需要配置mock框架,我们已经配置了。
2、测试类上加上RunWith的注解
3、要mock的对象加上@Mock注解
4、实际使用Mock的对象方法,需要替换返回值
@RunWith(MockitoJUnitRunner::class)
class ExampleUnitTest {
private val FAKE_STRING = "TestDemo"
@Mock
internal var mMockContext: Context? = null
@Test
fun mockTest() {
`when`(mMockContext!!.getString(R.string.app_name))
.thenReturn(FAKE_STRING)
val result = mMockContext!!.getString(R.string.app_name)
assertEquals(result, FAKE_STRING)
}
实际上本地的单元测试如果要用android相关的代码或者android环境下的数据是不可以的,我们仍可以mock出一个模拟的环境来达到自己的测试目的。
进一步的了解mockito,可以看官方文档。
构建设备单元测试
先配置好环境,添加需要的依赖。
android{
defaultConfig{
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
}
dependencies {
androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
exclude group: 'com.android.support', module: 'support-annotations'
})
}
使用test包下提供的各种方法获取当前的运行的环境等。例子中是获取了application的context,来获取app的包名。
@RunWith(AndroidJUnit4.class)
public class ExampleInstrumentedTest {
@Test
public void useAppContext() throws Exception {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getTargetContext();
assertEquals("com.example.administrator.testdemo", appContext.getPackageName());
}
}
然后点击运行到具体的设备上。最终绿桥通过。
创建测试套件
理解为测试的集合,可以将多个测试类集合到一起,运行这个套件即可。
@RunWith(Suite::class)
@Suite.SuiteClasses(ExampleInstrumentedTest::class, ExampleInstrumentedTest2::class)
class UnitTestSuite {
}
在两段测试代码中添加:System.out.println("test1")和System.out.println("test2")代码,最终运行这个套件,两段代码均被打印了出来。
使用Firebase Test Lab
Firebase是谷歌的测试实验室,使用的是google的数据中心内托管的设备。就是远程使用google的设备测试。谷歌的设备齐全,尺寸的兼容性,设备的兼容性都可以测出来。其实就是远程租用的模式。当然需要注册账号,配置应用,具体的使用可以去看官网的介绍。
类似的国内也有很多平台,不过google使用起来的很方便直接在android studio即可配置,比较方便。而且可以免费使用(不过有一些限制)。想用吗?前提是你需要科学上网。
后记
学会了怎么构建单测了,我们就开始在自己的项目中添加测试代码吧。