15

在测试中我们通常有assertNotNull,但它不会执行从可为空类型到非可空类型的智能转换。我必须写这样的东西:

if (test == null) {
    Assert.fail("")
    return
}

assertNotNull仅通过呼叫执行智能投射是否是一种解决方法?你如何解决?

4

3 回答 3

12

不幸的是,您调用的函数体(包括内联函数)不用于智能转换和可空性推断。

您的代码中没有太多可以改进的地方,我只建议一件事:您可以将 Elvis 运算符Nothing这些断言语句的函数一起使用。控制流分析考虑了导致的分支Nothing并从中推断出可空性:

fun failOnNull(): Nothing = throw AssertionError("Value should not be null")

val test: Foo? = foo()

test ?: failOnNull()
// `test` is not-null after that

这也可以在没有函数的情况下编写:test ?: throw AssertionError("..."),因为throw表达式也有类型Nothing


说到断言失败的更一般情况,人们可能会使用一个fail(...): Nothing函数,它也为控制流分析提供了额外的提示。JUnitAssert.fail(...)不是一个Nothing函数,但您可以在kotlin-test-junit模块中找到一个或编写自己的一个。

test as? SomeType ?: fail("`test` should be an instance of SomeType")
// smart cast works here, `test` is `SomeType`
于 2017-02-05T14:08:24.243 回答
3

kotlin.test库为此提供了一个简单的解决方案:

kotlin.test.assertNotNull()

因为这个函数实现了 Kotlin 合约,所以它支持智能转换:

contract { returns() implies (actual != null) }

例子:

    fun Foo?.assertBar() {
        assertNotNull(this)
        assertEquals(this.bar, 0)
    }

只要确保使用正确的assertNotNull导入(import kotlin.test.assertNotNull)!

如果您还没有使用该kotlin.test库,请将其添加到您的项目中:

group: 'org.jetbrains.kotlin', name: 'kotlin-test', version: '1.3.11

于 2019-01-22T13:59:01.770 回答
0

非常感谢@Rolf 在他的回答中将我指向kotlin -test库提供的assertNotNull方法。

我只想补充并指出,此方法具有非空返回类型(即,它返回作为非空对象传递给它的可空对象)。因此,如果您在测试中强制展开属性,您可以摆脱这种情况并改进您的测试,如下所示:

val myProperty = assertNotNull(myObject?.myProperty)
assertEquals("Foo", myProperty.someOtherProperty)
于 2020-11-18T11:05:53.273 回答