Котлин контракты: ссылка не-ноль из двух свойств
Скажем, у меня есть такой класс:
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 кажется способом рассказать компилятору об этих типах отношений. Возможен ли вышеуказанный инвариант для моделирования с контрактами Котлина?
Мой конечный результат состоит в том, чтобы позволить коду наподобие следующего компиляции:
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())
}
1 ответ
Решение
Это кажется невозможным в 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