Android应用测试开发——单元测试

前言

上一篇文章,我们了解了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(""))
    }
Android应用测试开发——单元测试_第1张图片
erro.png

对于这类的方法,最好的方案是重新写一个自己的TextUtils,使用自己的方法。或者测试用例不使用和android相关的代码。当然也是有绕过这个错误的。

android{
    ...
    testOptions {
        unitTests.returnDefaultValues = true
    }
}

不过最好不要这样做,这样会导致所有的这种类型都会返回默认值。TextUtils.isEmpty("haha")返回的结果也是true。


Android应用测试开发——单元测试_第2张图片
image.png
使用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")代码,最终运行这个套件,两段代码均被打印了出来。


Android应用测试开发——单元测试_第3张图片
image.png
使用Firebase Test Lab

Firebase是谷歌的测试实验室,使用的是google的数据中心内托管的设备。就是远程使用google的设备测试。谷歌的设备齐全,尺寸的兼容性,设备的兼容性都可以测出来。其实就是远程租用的模式。当然需要注册账号,配置应用,具体的使用可以去看官网的介绍。

Android应用测试开发——单元测试_第4张图片
config.png

类似的国内也有很多平台,不过google使用起来的很方便直接在android studio即可配置,比较方便。而且可以免费使用(不过有一些限制)。想用吗?前提是你需要科学上网。

后记

学会了怎么构建单测了,我们就开始在自己的项目中添加测试代码吧。

你可能感兴趣的:(Android应用测试开发——单元测试)