【问题标题】:How to use an android dependency in unit tests?如何在单元测试中使用 android 依赖项?
【发布时间】:2019-05-26 12:51:47
【问题描述】:

我目前正在尝试 TDD,但无法弄清楚如何在我的单元测试中使用 android 依赖项。

我想将 PONS API 的 JSON 响应解析为一个对象。 为了测试我的 Jsonparser,我尝试创建一个模拟间谍,或者只是为 JsonReader 的普通实例创建 android.utils.JsonReader,然后我将其传递给我的解析器。


@Mock
private var reader : JsonReader = JsonReader(BufferedReader(FileReader(fileName)))


@Before
fun setUp(){
    MockitoAnnotations.initMocks(this)
}

@Test
fun parseTest() {

    val response = ResponseParser.parse(reader)

    val entry : response.Entry = response.getEntry(0)

    assertThat(entry.strings_displayed[0].get(0).first, `is`("Haus (Wohnhaus, Heim)"))
    assertThat(entry.strings_displayed[0].get(0).second, `is`("casa"))
    }

但是我的尝试都没有产生所需的输出。截至目前,我无法使用 JsonReader 类。我得到的只是这个错误消息(原因已被谷歌搜索)

java.lang.RuntimeException: Method beginObject in android.util.JsonReader not mocked. See http://g.co/androidstudio/not-mocked for details.
at android.util.JsonReader.beginObject(JsonReader.java)
at com.huckebrink.remindyct.Data.PONSRequestParser$Companion.readJsonString(PONSRequestParser.kt:23)
at com.huckebrink.remindyct.Data.PONSRequestParser$Companion.parse(PONSRequestParser.kt:13)
at com.huckebrink.remindyct.ParsingUnitTest.parseTest(ParsingUnitTest.kt:47)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:26)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

【问题讨论】:

  • 如果被测类有android依赖,那么ypu应该模拟它们,你可以使用工具:mock/powermock/robolectric,如果你的类根本没有android依赖,那么只需创建对象并测试。
  • 也已经尝试过了。但是收到了同样的错误。
  • 你能把这个类用java测试一下吗,我稍后看看

标签: android unit-testing kotlin mockito


【解决方案1】:

好吧,我的建议是尽量保持您的单元测试级别而不依赖于 Android 框架,如果您只能在单元级别使用 Kotlin 进行测试,而将更复杂的测试留给集成级别,那么一切都会变得简单。

有时,如果应用程序是耦合的,这并不容易,因此,您可以使用一些库作为 Robolectric 让 Android 框架在您的单元测试中部分工作。

搜索:“Robolectric 单元测试”以获取示例。祝你好运

更新:

来自 Android 文档https://developer.android.com/training/testing/unit-testing

本地测试:仅在本地计算机上运行的单元测试。这些测试被编译为在 Java 虚拟机 (JVM) 上本地运行,以最大限度地减少执行时间。如果您的测试依赖于 Android 框架中的对象,我们建议使用 Robolectric。对于依赖于您自己的依赖项的测试,请使用模拟对象来模拟您的依赖项的行为。

更新 2:

JSON 解析应该是检测/集成级别测试的一部分,无论如何,如果您希望依赖项在测试目录上使用 JsonParing,您可以添加:

testCompile ‘org.json:json:20140107’

【讨论】:

  • 但是你真的应该使用这么重的库,只是为了测试我的 JSON 解析吗?使用 Roboelectric 它可以正常工作,但需要太长时间恕我直言
  • 这就是为什么像 JSON 解析这样的任务不是单元测试而是集成测试的原因。单元测试应该只在 JVM 上运行才能快速运行,而 Android 框架显然不是 JVM 的一部分。您有像 Robolectric 这样的替代品(由 Google 推荐),但是如果您阅读了有关测试最佳实践 (developer.android.com/training/testing) 的信息,您会发现一些测试非常适合单元级别(测试目录),而更复杂的测试更适合集成级别(androidTest 目录)。
  • 但是,如果您认为以这种方式进行单元测试对您的案例来说是个好主意,您可以添加:testCompile ‘org.json:json:20140107’
  • 嗯,它是:'testImplementation'。但这行得通。谢谢
  • 另外值得注意的是testImplementation 'com.beust:klaxon:5.0.1'
猜你喜欢
  • 1970-01-01
  • 2020-11-21
  • 1970-01-01
  • 2021-05-21
  • 1970-01-01
  • 2014-05-29
  • 2012-07-15
  • 2012-07-11
  • 1970-01-01
相关资源
最近更新 更多