[关闭]
@cxm-2016 2016-12-25T20:54:37.000000Z 字数 2796 阅读 2240

Kotlin (二十)——内联函数

Kotlin

版本:1
作者:李颖


内联函数(Inline Function)

使用 高阶函数 在运行时会带来一些不利: 每个函数都是一个对象, 而且它还要捕获一个闭包, 也就是, 在函数体内部访问的那些外层变量.
内存占用(函数对象和类都会占用内存) 以及虚方法调用都会带来运行时的消耗.

但在很多情况下, 通过将 Lambda 表达式内联在使用处, 可以消除这些运行时消耗.
上文中的那些函数就是很好的例子. 也就是说, lock() 函数可以很容易地内联在调用处.
看看下面的例子:

  1. lock(l) { foo() }

编译器可以直接产生下面的代码, 而不必为参数创建函数对象, 然后再调用这个参数指向的函数:

  1. l.lock()
  2. try {
  3. foo()
  4. }
  5. finally {
  6. l.unlock()
  7. }

这不就是我们最初期望的东西吗?

为了让编译器做到这点, 我们需要使用 inline 修饰符标记 lock() 函数:

  1. inline fun lock<T>(lock: Lock, body: () -> T): T {
  2. // ...
  3. }

inline 修饰符既会影响到函数本身, 也影响到传递给它的 Lambda 表达式: 这两者都会被内联到调用处.

函数内联也许会导致编译产生的代码尺寸变大, 但如果我们使用合理(不要内联太大的函数), 可以换来性能的提高, 尤其是在循环内发生的 "megamorphic" 函数调用. (译注: megamorphic 意义不明)

noinline

如果一个内联函数的参数中有多个 Lambda 表达式, 而你只希望内联其中的一部分, 你可以对函数的一部分参数添加 noinline 标记:

  1. inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) {
  2. // ...
  3. }

可内联的 Lambda 表达式只能在内联函数内部调用, 或者再作为可内联的参数传递给其他函数, 但 noinline 的 Lambda 表达式可以按照我们喜欢的方式任意使用: 可以保存在域内, 也可以当作参数传递, 等等.

注意. 如果一个内联函数不存在可以内联的函数类型参数, 而且没有 实体化的类型参数, 编译器将会产生一个警告, 因为将这样的函数内联不太可能带来任何益处(如果你确信需要内联, 可以关闭这个警告).

非局部返回(Non-local return)

在 Kotlin 中, 使用无限定符的通常的 return 语句, 只能用来退出一个有名称的函数, 或匿名函数.
这就意味着, 要退出一个 Lambda 表达式, 我们必须使用一个 标签, 无标签的 return 在 Lambda 表达式内是禁止使用的, 因为 Lambda 表达式不允许强制包含它的函数返回:

  1. fun foo() {
  2. ordinaryFunction {
  3. return // 错误: 这里不允许让 `foo` 函数返回
  4. }
  5. }

但是, 如果 Lambda 表达式被传递去的函数是内联函数, 那么 return 语句也可以内联, 因此 return 是允许的:

  1. fun foo() {
  2. inlineFunction {
  3. return // OK: 这里的 Lambda 表达式是内联的
  4. }
  5. }

这样的 return 语句(位于 Lambda 表达式内部, 但是退出包含 Lambda 表达式的函数) 成为 非局部(non-local) 返回. 我们在循环中经常用到这样的结构, 而循环也常常就是包含内联函数的地方:

  1. fun hasZeros(ints: List<Int>): Boolean {
  2. ints.forEach {
  3. if (it == 0) return true // 从 hasZeros 函数返回
  4. }
  5. return false
  6. }

注意, 有些内联函数可能并不在自己的函数体内直接调用传递给它的 Lambda 表达式参数, 而是通过另一个执行环境来调用, 比如通过一个局部对象, 或者一个嵌套函数. 这种情况下, 在 Lambda 表达式内, 非局部的控制流同样是禁止的. 为了标识这一点, Lambda 表达式参数需要添加 crossinline 修饰符:

  1. inline fun f(crossinline body: () -> Unit) {
  2. val f = object: Runnable {
  3. override fun run() = body()
  4. }
  5. // ...
  6. }

在内联的 Lambda 表达式中目前还不能使用 breakcontinue, 但我们计划将来支持它们

实体化的类型参数(Reified type parameter)

有些时候我们需要访问作为参数传递来的类型:

  1. fun <T> TreeNode.findParentOfType(clazz: Class<T>): T? {
  2. var p = parent
  3. while (p != null && !clazz.isInstance(p)) {
  4. p = p?.parent
  5. }
  6. @Suppress("UNCHECKED_CAST")
  7. return p as T
  8. }

这里, 我们向上遍历一颗树, 然后使用反射来检查节点是不是某个特定的类型. 这些都没问题, 但这个函数的调用代码不太漂亮:

  1. myTree.findParentOfType(MyTreeNodeType::class.java)

我们真正需要的, 只是简单地将一个类型传递给这个函数, 也就是说, 象这样调用它:

  1. myTree.findParentOfType<MyTreeNodeType>()

为了达到这个目的, 内联函数支持 实体化的类型参数(reified type parameter), 使用这个功能我们可以将代码写成:

  1. inline fun <reified T> TreeNode.findParentOfType(): T? {
  2. var p = parent
  3. while (p != null && p !is T) {
  4. p = p?.parent
  5. }
  6. return p as T
  7. }

我们给类型参数添加了 reified 修饰符, 现在, 它可以在函数内部访问了, 就好象它是一个普通的类一样. 由于函数是内联的, 因此不必使用反射, 通常的操作符, 比如 !isas 都可以正常工作了. 此外, 我们可以象前面提到的那样来调用这个函数: myTree.findParentOfType<MyTreeNodeType>().

虽然很多情况下并不需要, 但我们仍然可以对一个实体化的类型参数使用反射:

  1. inline fun <reified T> membersOf() = T::class.members
  2. fun main(s: Array<String>) {
  3. println(membersOf<StringBuilder>().joinToString("\n"))
  4. }

通常的函数(没有使用 inline 标记的) 不能够使用实体化的类型参数.
一个没有运行时表现的类型(比如, 一个没有实体化的类型参数, 或者一个虚拟类型, 比如 Nothing) 不可以用作实体化的类型参数.

关于实体化类型参数的更底层的介绍, 请参见 规格文档.

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注