1

假设我有这样的课程:

data class URLAndPath(
   val baseUrl: URL,
   val path: String?
) {
    val url get(): URL? =
        try { path?.let { URL(baseUrl, it) } }
        catch(_: Exception) { null }

    init { require(path == null || url != null) { "Invalid URL $baseUrl$path" } } 
}

此类确保path != null当且仅当url != null

Kotlin 契约似乎是告诉编译器这些关系的一种方式。上述不变量是否可以用 Kotlin 合约建模?

我的最终结果是让如下代码编译:

val x = URLAndPath(URL("http://example.org/"), "index.html")
if(x.path != null) {
    // currently: Error: Only safe (?.) or non-null asserted (!!.) calls are allowed on a nullable receiver of type URL?
    println(x.url.toURI())
}
4

1 回答 1

2

这在 Kotlin 1.3 中似乎是不可能的,因为合约只能在顶级函数上,而不是在方法上。

例如

@ExperimentalContracts
data class NullableString(val s: String?) {

    fun isNotNull(): Boolean {
        contract {
            returns(true) implies (this@path != null)
        }
        return path != null
    }
}

不编译:

Error:(16, 8) Contracts are allowed only for top-level functions
Error:(17, 39) Unresolved reference: @path
Error:(19, 15) Unresolved reference: path
于 2018-12-12T18:50:59.473 回答