关于Kotlin语法的异议

原文发表于2017-09-01。

情况

Lambda的表示法是{ ... } ,例如:

val func = {
  println()
}
val func = { x ->
  println(x)
}

若函数的唯一或最后一个参数是函数类型,可以不需要用括号围住这个参数,这样就能随手写出这样漂亮的DSL:

// transaction(...)接受一个类型为函数的参数
// Auto handle a transaction
transaction {
  saveData()
}

// use(...)接受一个类型为函数的参数
// Auto close the resource
inputStream.use {
  consume(it)
}

但是就不能像C/Java/Scala一样用花括号把代码组织成普通代码块了,而是必须调用run函数,这是Kotlin开发组的一种取舍:

// Java
String externalName;
{
  String internalName = getInternalName();
  externalName = convert(internalName);
}
// Scala
val externalName = {
  val internalName = getInternalName()
  convert(internalName)
}
// Kotlin
val externalName = run {
  val internalName = getInternalName()
  convert(internalName)
}

run函数会被编译器内联,没有函数调用的额外开销,只是语法上与C家族不一致,而且有点繁琐。

类似run的函数有好几个,我们来瞧一瞧。

run(block: () -> R): R = block()

执行block,返回block的结果R,相当于Scala的普通代码块

T.run(block: T.() -> R): R = block()

把T作为this,执行block,返回block的结果R

T.let(block: (T) -> R): R = block(this)

把T作为block的入参(it),执行block,返回block的结果R

T.apply(block: T.() -> Unit): T { block(); return this }

把T作为this,执行block,返回T本身

T.also(block: (T) -> Unit): T { block(this); return this }

把T作为block的入参(it),执行block,返回block的结果R

with(receiver: T, block: T.() -> R): R = receiver.block()

把第一个参数作为this传给block。能表达这种代码:with(entry) { consumeKeyValue(key, value) }

当我们想随手转换一个对象,可以这么写:

val userDTO = user.run {
  UserDTO(name, password)
}

或这么写

val userDTO = user.let {
  UserDTO(it.name, it.password)
}

当我们新建了一个对象,想随手给它完成初始化,可以这么写:

val user = User().apply {
  name = "Mr Wang"
  password = "&%&&**("
}

或这么写

val user = User().let {
  it.name = "Mr Wang"
  it.password = "&%&&**("
}

你可能觉得这只是微小的语法糖,那么看这个nullable的例子:

// 繁琐写法
val userDTO =
  if (user != null) {
    UserDTO(user.name, user.password)
  } else {
    null
  }
// 简洁写法
val userDTO = user?.run {
  UserDTO(name, password)
}

是不是高下立见?

批评

为了支持多种写法,占用了很多名字。像run这种在JDK中常见的名字也被用了,虽然有静态检查,但同一个名字被安上不同的语义仍然是一种心智负担。况且run这个名字完全没有表达“转换”的意思嘛!

使用函数之前要先想好接下来的写法适合哪个名字的函数,也是一种心智负担。

为了使语义更明确,我建议做如下改进:

减少内置函数名:去掉run和also,只保留let,apply和with。采用如下几种函数签名:

let(block: () -> R)

T.let(block: T.() -> R)
T.let(block: (T) -> R): R

T.apply(block: T.() -> Unit): T
T.apply(block: (T) -> Unit): T

with(receiver: T, block: T.() -> R): R

如上,with没有变,run被并入let,also被并入apply。

let表示定义新的变量,apply表示对现有变量做一些处理,with表示以现有变量作为scope来做一些事(包括返回新的变量)。语义很清楚。

let总是返回结果R,apply总是返回原本的T。let/apply若接受无参函数,就把T作为this,若接受唯一参数为T的函数,就把T作为参数传入,不允许隐式的it参数。

// 使用this
user.apply {
  name = "Mr Wang"
  password = "&%&&**("
}

// 使用it必须显式声明
user.apply { it ->
  it.name = ""
  it.password = "&%&&**("
}

另一个问题,从Java转过来的应用开发者可能会习惯性地写一个代码块,忘了这实际上是一个Lambda,是不会执行的。

// 不会执行
{
  doSomething()
}

// 会执行
{
  doSomething()
}()

原则上应禁止定义未被使用的Lambda,以免误写出永远不会被执行的代码。

posted @ 2020-12-25 18:18  计算法  阅读(60)  评论(0编辑  收藏  举报