当您测试元素或元素系统时,您是在隔离状态下进行的。例如,要测试 ViewModel,您不需要启动模拟器并启动 UI,因为它不依赖(或不应该依赖)Android 框架。
但是,被测对象可能依赖于其他对象才能工作。例如,ViewModel 可能依赖于数据存储库才能工作。
当您需要向被测对象提供依赖项时,一种常见的做法是创建测试替身(或测试对象)。测试替身是看起来和行为都像应用中组件的对象,但它们是在您的测试中创建的,以提供特定的行为或数据。主要优点是它们使您的测试更快、更简单。
测试替身类型
有各种类型的测试替身
伪造 | 一个测试替身,它具有类的“工作”实现,但它的实现方式使其适合测试但不适合生产。 示例:内存数据库。 伪造不需要模拟框架,并且很轻量级。它们是首选。 |
---|---|
模拟 | 一个测试替身,它按照您编程的方式运行,并且对它的交互有期望。如果模拟的交互不符合您定义的要求,则模拟将导致测试失败。模拟通常使用模拟框架来实现所有这些。 示例:验证数据库中的方法是否恰好被调用了一次。 |
存根 | 一个测试替身,它按照您编程的方式运行,但对它的交互没有期望。通常使用模拟框架创建。为了简单起见,伪造优于存根。 |
虚拟 | 一个测试替身,它被传递但未使用,例如,如果您只需要将其作为参数提供。 示例:作为点击回调传递的空函数。 |
间谍 | 真实对象的包装器,它还跟踪一些其他信息,类似于模拟。通常避免使用它们来增加复杂性。因此,伪造或模拟优于间谍。 |
影子 | Robolectric 中使用的伪造。 |
使用伪造的示例
假设您想单元测试一个依赖于名为UserRepository
的接口并将第一个用户的名称公开给 UI 的 ViewModel。您可以通过实现接口并返回已知数据来创建伪造的测试替身。
object FakeUserRepository : UserRepository {
fun getUsers() = listOf(UserAlice, UserBob)
}
val const UserAlice = User("Alice")
val const UserBob = User("Bob")
此伪造的UserRepository
不需要依赖于生产版本将使用的本地和远程数据源。该文件位于测试源集中,不会与生产应用一起发布。
以下测试验证 ViewModel 是否正确地将第一个用户名公开给视图。
@Test
fun viewModelA_loadsUsers_showsFirstUser() {
// Given a VM using fake data
val viewModel = ViewModelA(FakeUserRepository) // Kicks off data load on init
// Verify that the exposed data is correct
assertEquals(viewModel.firstUserName, UserAlice.name)
}
在单元测试中用伪造替换UserRepository
很容易,因为 ViewModel 是由测试人员创建的。但是,在更大的测试中替换任意元素可能具有挑战性。
替换组件和依赖项注入
当测试无法控制被测系统的创建时,替换测试替身的组件会变得更加复杂,并且需要您的应用架构遵循可测试的设计。
即使是大型端到端测试也可以从使用测试替身中获益,例如在应用中遍历完整用户流程的检测 UI 测试。在这种情况下,您可能希望使您的测试独立。独立测试避免所有外部依赖项,例如从互联网获取数据。这提高了可靠性和性能。
您可以手动设计您的应用以实现这种灵活性,但我们建议使用 依赖注入 框架(例如 Hilt)在测试时替换应用中的组件。请参阅 Hilt 测试指南。
后续步骤
在 测试策略 页面中,您可以了解如何使用不同类型的测试来提高工作效率。