news 2026/4/17 12:36:52

Android单元测试

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Android单元测试

Android单元测试基础

单元测试用于验证应用中最小单元(函数或类)的行为是否正确。在 Android/Kotlin 项目中,本地单元测试通常放在module/src/test/目录下,使用 JUnit4 框架编写。要启用测试,需要在 Gradle 中添加依赖,例如

testImplementation "junit:junit:版本号"(JUnit4)和testImplementation "io.mockk:mockk:版本号"(MockK)。

每个测试类中包含一个或多个用@Test注解标记的方法,在方法体内调用被测函数并使用断言验证输出。

  1. class EmailValidatorTest {

  2. @Test

  3. fun emailValidator_CorrectEmailSimple_ReturnsTrue() {

  4. assertTrue(EmailValidator.isValidEmail("name@example.com"))

  5. }

  6. }

该示例中,测试方法通过assertTrue检查isValidEmail()的返回值。在编写单元测试时,我们通常将外部依赖(如数据库、网络、Android 框架等)替换为可控的测试替身(如 mock 对象),以实现隔离测试。常用的断言库包括 JUnit Assert、Hamcrest、Truth 等。

测试替身

单元测试中的测试替身(Test Doubles)

依赖隔离,核心逻辑:

关键概念解析
替身类型作用场景示例
Mock验证交互行为检查是否调用了数据库API
Stub返回预设数据固定返回用户{id:1, name:测试}
Fake简化实现替代真实服务内存数据库替代MySQL
Spy记录调用信息(Mock的变体)记录网络请求次数
Dummy填充参数(无逻辑)new Object()占位
在Android测试中的典型应用
  1. // 使用Mockito框架示例

  2. @Mock

  3. Database mockDB; // 创建数据库Mock

  4. @Test

  5. public void testUserSave() {

  6. // 1. 设置Stub行为

  7. when(mockDB.save(any(User.class))).thenReturn(true);

  8. // 2. 执行被测方法

  9. service.saveUser(new User("test"));

  10. // 3. 验证Mock交互

  11. verify(mockDB).save(any(User.class));

  12. }

核心价值
  • 🛡️隔离性:避免测试因网络/DB故障而失败
  • 加速测试:移除真实IO操作(原需200ms→2ms)
  • 🔍行为验证:确保调用次数/参数符合预期
  • 🧪边界覆盖:轻松模拟异常场景(如:when(...).thenThrow(...)

MockK 概念与常用用法

MockK 是一个专为 Kotlin 设计的模拟测试框架,相比 Mockito 等 Java 库,MockK 自然支持 Kotlin 的特性,如 final 类、扩展函数和协程。使用 MockK 可以方便地创建接口或类的 mock 对象,并通过 DSL 定义其行为。最简单的使用方法如下:

  1. val car = mockk<Car>() // 创建 Car 类的 mock 对象

  2. every { car.drive(Direction.NORTH) } returns Outcome.OK // 定义方法返回值

  3. car.drive(Direction.NORTH) // 调用时返回 OK

  4. verify { car.drive(Direction.NORTH) } // 验证方法被调用

  5. confirmVerified(car)

以上示例来自 MockK 官方文档。其中mockk<T>()会创建一个严格模式(strict)的 mock 对象,需要显式定义其所有行为(使用every { ... } returns ...)。

什么叫做严格模式
1.什么是严格模式?
  • 在 MockK 中,严格模式(也称为标准模式)意味着 mock 对象会严格执行以下规则:
    • 所有对 mock 对象方法的调用都必须预先声明(即使用every块定义行为)。
    • 如果调用了一个没有预先声明的方法,MockK 会立即抛出异常(MockKException: no answer found)。

例如:

  1. val car = mockk<Car>() // 创建严格模式的 mock 对象

  2. // 未定义行为时调用方法 → 抛出异常!

  3. car.drive(50) // 抛出 no answer found 异常

2.显式定义行为

使用every { ... } returns ...结构为 mock 对象的方法定义行为:

every { car.drive(50) } returns "Driving at 50 km/h"
  • every:声明一个预期调用的行为。
  • returns:指定该方法调用的返回值。

此时调用car.drive(50)会返回"Driving at 50 km/h"

3.为何需要显式定义所有行为?
  • 避免隐藏错误:严格模式强制测试编写者明确指定 mock 对象的所有预期行为。这有助于暴露测试中的隐含假设或遗漏的依赖调用。
  • 提高测试可靠性:确保测试只关注预先定义的行为,避免因意外调用导致的假阳性/假阴性结果。
4.未定义行为的后果

如果在严格模式下调用未定义的方法,会收到如下错误:

io.mockk.MockKException: no answer found for: Car(#1).drive(50)
5.对比:严格模式 vs 松弛模式
模式是否需要显式定义行为未定义行为时的处理
严格模式抛出异常
松弛模式返回默认值(null, 0 等)

松弛模式的创建方式:

val relaxedCar = mockk<Car>(relaxed = true) // 不会因未定义行为抛出异常
6.何时使用严格模式?
  • 推荐场景
    1. 需要精确控制 mock 行为的测试。
    1. 验证代码是否按预期调用了特定方法(通常结合verify)。
  • 不推荐场景:当 mock 对象有许多不重要的方法(如纯数据模型),使用严格模式会写大量every块,此时可改用松弛模式。
7.完整示例
  1. // 定义类

  2. class Car {

  3. fun drive(speed: Int): String = "Real driving: $speed km/h"

  4. }

  5. // 测试

  6. @Test

  7. fun testStrictMock() {

  8. // 1. 创建严格模式 mock

  9. val car = mockk<Car>()

  10. // 2. 显式定义行为

  11. every { car.drive(50) } returns "Mocked driving at 50 km/h"

  12. // 3. 调用已定义方法 → 成功

  13. assertEquals("Mocked driving at 50 km/h", car.drive(50))

  14. // 4. 调用未定义方法 → 抛出异常!

  15. assertFailsWith<MockKException> {

  16. car.drive(100) // 未定义 100 的行为

  17. }

  18. }

总结
  • mockk<T>()创建的是严格模式的 mock 对象。
  • 必须every { ... } returns ...为其每个需要调用的方法定义行为。
  • 严格模式能提高测试的精确性,但会增加样板代码。根据场景选择是否使用松弛模式(relaxed = true)。### 详细解释:mockk<T>()创建严格模式(Strict Mode)的 Mock 对象
核心概念:严格模式 (Strict Mode)
val myService = mockk<MyService>() // 创建严格模式的 mock 对象
  • 行为必须显式定义

    • 在严格模式下,mock 对象的所有交互行为都必须预先声明
    • 如果调用了未定义的方法,会立即抛出异常:
      io.mockk.MockKException: no answer found for: MyService(#1).getData()
  • 定义行为的方式
    使用every { ... } returns ...结构显式声明行为:

  1. // 必须为每个需要调用的方法定义行为

  2. every { myService.getData() } returns "MockedData"

  3. every { myService.calculate(any()) } returns 42

为什么需要严格模式?
场景严格模式非严格模式
未定义方法调用立即抛出异常返回默认值(null, 0 等)
测试可靠性确保不会意外调用未模拟的方法可能隐藏未覆盖的依赖
测试意图明确声明所有预期行为隐含接受默认行为

典型错误示例
  1. // 测试代码

  2. val userService = mockk<UserService>() // 严格模式

  3. // ❌ 忘记定义行为

  4. userService.findUser("id123") // 抛出 MockKException!

  5. // ✅ 正确做法

  6. every { userService.findUser(any()) } returns User("MockedUser")

  7. val result = userService.findUser("id123") // 返回 User 对象

严格模式的核心特点
  1. 零容忍未声明行为
    任何未通过every定义的调用都会导致测试失败。

  2. 精确控制模拟行为
    必须为每个参数组合指定行为:

    1. // 不同参数需要单独定义

    2. every { myService.parse("A") } returns 1

    3. every { myService.parse("B") } returns 2

    与验证的配合
    常与verify一起使用确保调用符合预期:

    1. every { myService.send(any()) } returns true

    2. // 测试代码

    3. myService.send("message")

    4. verify { myService.send("message") } // 验证调用发生

何时使用严格模式?
  • 推荐场景

    • 需要精确控制依赖行为的测试
    • 验证复杂交互逻辑
    • 关键服务/组件的测试
  • 替代方案(非严格模式)

val relaxedService = mockk<MyService>(relaxed = true) // 允许未定义调用

Mock 接口、类、静态和扩展方法

  • 接口/类的 mock:对于普通的接口或类,使用mockk<类型>()创建 mock 对象。例如val repo = mockk<MyRepository>()。注意 MockK 能直接 mock Kotlin 中的final class,无需像 Mockito 那样特殊配置。也可以使用注解@MockK并在测试@Before中调用MockKAnnotations.init(this)或使用前述的MockKRule进行初始化。

  • Relaxed Mock:如果不想为每个方法都定义返回值,可以创建一个relaxedmock,即mockk<MyClass>(relaxed = true)。这会让所有非 Unit 返回类型的方法自动返回默认值(例如布尔型为 false,引用型为 null)。这样即使不显式 stub 方法,调用时也不会抛异常。需要注意,针对泛型返回类型的函数,放宽 mock 有时可能抛出类型转换异常。

  • 静态方法和顶层函数:Kotlin 的顶层函数或 Java 静态方法可以用mockkStatic()模拟。对于 Java 静态方法或类静态方法,直接传入类引用:

  1. mockkStatic(Uri::class)

  2. every { Uri.parse("http://test/path") } returns mockUri

这会拦截Uri.parse()的调用,返回自定义结果。在模拟 Kotlin 顶层(module-wide)函数或扩展函数时,可以传入生成的类名字符串(通常是包名+文件名+Kt后缀)或函数引用。例如文档中示例,将扩展函数所在文件File.kt(包名为 pkg)中所有函数静态化:

  1. mockkStatic("pkg.FileKt")

  2. every { Obj(5).extensionFunc() } returns 11

  • 如此可以模拟Obj.extensionFunc()的行为。总之,mockkStatic适用于替换任何静态或顶层函数的实现,模拟完成后可用unmockkStatic解除

  • 对象(单例)的 mock:对于 Kotlin 的object或 Java 的单例,可使用mockkObject(SomeObject)创建 mock 对象。此时可以像普通 mock 一样用every { ... }定义行为,并在测试后调用unmockkObject(SomeObject)恢复原状。

  • 注意事项:使用mockkStaticmockkObject后要记得在测试结束时使用unmockkStaticunmockkObject清理,否则可能影响后续测试。

感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 6:46:18

软件测试经典面试题

问&#xff1a;网页字符统计功能如何测试&#xff1f;测试点有哪些&#xff1f; &#xff08;例&#xff1a;计算一个文本字符串中a出现的个数&#xff09; 一、核心功能测试点&#xff08;验证基础逻辑&#xff09; 基础计数准确性 单字符输入&#xff08;如 "a"&…

作者头像 李华
网站建设 2026/4/18 11:05:43

三大视觉大模型对比:Glyph/Qwen-VL/Llama3部署评测

三大视觉大模型对比&#xff1a;Glyph/Qwen-VL/Llama3部署评测 1. 视觉大模型的现实挑战与新思路 你有没有遇到过这样的问题&#xff1a;想让AI读完一篇上万字的技术文档&#xff0c;结果它只记得最后一段&#xff1f;传统语言模型受限于上下文长度&#xff0c;处理长文本时要…

作者头像 李华
网站建设 2026/4/18 1:39:56

Z-Image-Turbo降本部署案例:低成本GPU方案实操手册

Z-Image-Turbo降本部署案例&#xff1a;低成本GPU方案实操手册 你是否也在为图像生成模型的高昂部署成本头疼&#xff1f;动辄需要A100、H100这类高端显卡&#xff0c;让很多个人开发者和中小团队望而却步。今天要分享的这个实战案例&#xff0c;可能会让你眼前一亮——我们用…

作者头像 李华
网站建设 2026/4/17 22:56:00

PHP版本性能大比拼(从7.4到8.4):实测数据告诉你提升到底有多少

第一章&#xff1a;PHP版本性能大比拼的背景与意义 在现代Web开发中&#xff0c;PHP作为最广泛使用的服务器端脚本语言之一&#xff0c;其版本迭代对应用性能有着深远影响。不同PHP版本在底层引擎优化、内存管理机制和执行效率方面存在显著差异&#xff0c;直接影响网站响应速度…

作者头像 李华