Kotlin 使用 lambda 作为初始化器

Kotlin using lambda as initializer

所以我有以下困境。这不是关于什么东西不起作用的问题,而是关于什么东西会更好 elegant/good 练习以及为什么。

所以,我们知道 init 块,其中包含初始化逻辑。也许你打开一个文件或从配置文件中读取一些常量,也许你根据一些使用构造函数参数的更复杂的算法设置了一些属性。

但是,如果您的 one/some 属性确实需要一点逻辑,一些琐碎的逻辑,例如验证参数或类似的事情。而且这个逻辑只与那个特定的属性相关,与其他属性没有交互?

例如:

class MyCircularQueue(k: Int) {

    private val arr = { 
        if (k < 1) 
            throw IllegalArgumentException("k must be at least 1") 
        else  
            Array(k) { 0 }
    }()

    private var head = 0
    private var tail = 0
    private var empty = true

}

这里arr需要初始化为0的数组,但是如果k小于1显然有问题,需要很短的初始化逻辑,只是一个抛出异常或初始化数组的简单检查。没有与其他属性的交互,没有复杂的逻辑,只有极其琐碎的检查或逻辑。

在这种情况下,我想知道将琐碎的初始化逻辑放在 "makeshift" 代码块(现场调用的 lambda)中是否会更好。好处是声明和初始化逻辑将在同一个地方,而不是不必要地分开,我认为这样它更具可读性。

所以在我写完这篇文章之后,我想知道是否还有其他人因为非常简单的初始化逻辑而遇到这个问题,但我真的找不到任何相关信息。

所以底线:这样可以吗?是好的做法,还是至少 不是 的坏做法?是否有意义?在 Kotlin 中有更好的 and/or 官方方法吗?

不需要函数变通,if是一个表达式,returns是一个值,所以可以简化为:

private val arr =
    if (k < 1)
        throw IllegalArgumentException("k must be at least 1")
    else
        Array(k) { 0 }

另外,Kotlin 在标准库中附带了很多函数式 API,可以帮助创建简洁的单行代码。在这种情况下,您可以这样做:

private val arr =
     k.takeIf { it > 0 }?.let { Array(k) { 0 } } ?: throw IllegalArgumentException("k must be at least 1")

正如 s1m0nw1 的回答所说,您在这里并不真正需要它,但是当您需要时它会很有用,例如初始化程序中的局部变量。

在这种情况下,我不会使用 () 来调用 lambda,而是使用 run 来很好地内联它并且更明显,例如

private val arr = run {
  val temp1 = ...
  val temp2 = ...
  arrayOf(temp1, temp2)
}