Kotlin基础之内联函数

简介:

内联函数

使用高阶函数会给运行时带来一些坏处:每个函数都是一个对象,捕获闭包(如:访问函数体内的变量),内存分配(函数对象或Class),虚拟调用引入的运行过载。 使用内联Lambda表达式在多数情况下可以消除这种过载。比如下面的函数就是这种情况下的很好的例子,lock()函数可以很容易地在调用点进行内联扩展。

lock(l){ foo() }

编译能够产生下面的代码,而不是创建一个函数对象参数,生成调用。

l.lock()
try {
    foo()
}
finally {
    l.unlock()
}

也是我们一开始想要的。 为了让编译器能够这样执行,需要用inline修饰符来标记lock函数。

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

inline修饰符既影响函数对象本身,也影响传入的Lambda参数:两者都会被内联到调用点。

编译预处理器对内联函数进行扩展,省去了参数压栈、生成汇编语言的CALL调用、返回参数、执行return等过程,从而提高了运行速度。 使用内联函数的优点,在函数被内联后编译器就可以通过上下文相关的优化技术对结果代码执行更深入的优化。 内联不是万能药,它以代码膨胀为代价,仅仅省去了函数调用的开销,从而提高程序的执行效率。

说明:函数调用开销并不包括执行函数体所需要的开销,而是仅指参数压栈、跳转、退栈和返回等操作。如果执行函数体内代码的时间比函数调用的开销大得多,那么内联函数的效率收益会笑很多。另一方面每一处内联函数的调用都要拷贝代码,将使程序的总代码增大、消耗更多的内存空间。

noinline

如果只需要在内联函数中内联部分Lambda表达式,可以使用noinline来标记不需要内联的参数。

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

内联Lambda只能在内联函数中调用或作为内联参数,但noinline的Lambda可随意使用。
说明:没有内联函数参数和reified type parameters的内联函数,编译器会发出警告,因为内联这样的函数不见得有好处。

非局部返回

在Kotlin中可以使用正常、无条件的return退出有名和匿名函数,也意味需要使用一个标签来退出Lambda,在Lambda中禁止使用赤裸return语句,因为Lambda不能够使闭合函数返回。

fun foo(){
    ordinaryFunction{
        
        return // ERROR: can not make `foo` return here
    }
}

如果Lambda传入内联函数,则返回也是被内联,所以被允许。

fun foo(){
    inlineFunction {
        return // OK: the lambda is inlined
    }
}

这样的return(位于在Lambda中,但能够退出闭合函数)被称为非局部返回。Kotlin使用这种构造在有循环条件的闭合内联函数中。

fun hasZeros(ints: List<Int>): Boolean{
    ints.forEach{
        if(it == 0) return true // returns from hasZeros
    }
    
    return false
}

一些内联函数可能不是从函数体中直接调用传入的Lambda参数,而是从其他的执行上下文,如本地对象或嵌套函数。在这些情况下,non-local 控制流则不允许出现在Lambda中。使用crossinline修饰符来标记。

inline fun f(crossinline body: () -> Unit) {
    val f = object: Runnable {
        override fun run() = body()
    }
    // ...
}

具体化类型参数

有时需要访问传入函数中参数的类型。例如:

fun <T> TreeNode.findParentOfType(clazz: Class<T>): T? {
    var p = parent
    while (p != null && !clazz.isInstance(p)) {
        p = p.parent
    }
    @Suppress("UNCHECKED_CAST")
    return p as T?
}

在上述代码中,沿着树结构,使用反射来检查节点是否有指定类型。

treeNode.findParentOfType(MyTreeNode::class.java)

实际上想要只是简单给函数传入一个类型,如:

treeNode.findParentOfType<MyTreeNode>()

内联函数支持具体化参数类型,因此可以这样写:

inline fun <reified T> TreeNode.findParentOfType(): T? {
    var p = parent
    while (p != null && p !is T) {
        p = p.parent
    }
    return p as T?
}

使用reified修饰符限制参数类型,可以在内联函数中访问,就像是普通的Class。因为函数是内联的,不在需要反射,像!is和as的普通操作符执行。也可以像上述说的那样调用。

myTree.findParentOfType<MyTreeNodeType>()

尽管反射在很多情况不需要,仍需要使用它来具体话参数类型。

inline fun <reified T> membersOf() = T::class.members

fun main(s: Array<String>) {
    println(membersOf<StringBuilder>().joinToString("\n"))
}

内联属性

inline修饰符可以用在没有Backing Filed属性的访问函数。可以注解单独属性的访问函数。

val foo: Foo
    inline get() = Foo()

var bar: Bar
    get() = ...
    inline set(v) { ... }

甚至可以注解整个属性,让属性访问函数都变为内联函数。

inline var bar: Bar
    get() = ...
    set(v) { ... }

在调用时,内联访问函数与常规内联函数调用方式一样。

目录
相关文章
|
Java Kotlin
Kotlin内联函数inline、noinline、crossinline
如果一个函数接收另一个函数作为参数,或返回类型是一个函数类型,那么该函数被称为是高阶函数
131 0
|
Java Kotlin
Kotlin学历之内联函数
Kotlin学历之内联函数
78 0
Kotlin学历之内联函数
|
Java Kotlin
Kotlin中内联函数的作用是什么?
在以前,因为学过一段时间Kotlin(并没有实际开发中用过),很多东西都忘记了,但是kotlin的代码看起来其实和Java没什么区别,感觉都差不多。所以不要认为 Kotlin 很难学。
170 0
|
XML Java Android开发
kotlin 之几个常见的内联函数(六)—— 总结篇
本篇内容对内联函数做了一个总结,分别是let、with、run、apply、also
|
Java Kotlin
kotlin 之几个常见的内联函数(五)
适用于let函数的任何场景,also函数和let很像,只是唯一的不同点就是let函数最后的返回值是最后一行的返回值而also函数的返回值是返回当前的这个对象。一般可用于多个扩展函数链式调用。
141 0
|
XML Java 数据格式
kotlin 之几个常见的内联函数(四)
从结构上来看apply函数和run函数很像,唯一不同点就是它们各自返回的值不一样,run函数是以闭包形式返回最后一行代码的值,而apply函数的返回的是传入对象的本身。
111 0
|
Java Kotlin
kotlin 之几个常见的内联函数(三)
适用于let,with函数任何场景。因为run函数是let,with两个函数结合体,准确来说它弥补了let函数在函数体内必须使用it参数替代对象,在run函数中可以像with函数一样可以省略,直接访问实例的公有属性和方法,另一方面它弥补了with函数传入对象判空问题,在run函数中可以像let函数一样做判空处理
117 0
|
Java Android开发 Kotlin
kotlin 之几个常见的内联函数(二)
with函数它不是以扩展的形式存在的。它是将某对象作为函数的参数,在函数块内可以通过 this 指代该对象。同时this也可以省略,也就是在函数快里面可以直接使用对象的属性活方法。返回值为函数块的最后一行或指定return表达式。
122 0
|
Java Android开发 Kotlin
kotlin 之几个常见的内联函数(一)
kotlin现在是Android 开发的主流语言,因为代码简洁,特别是一些内联函数的使用,跟java的比起来,代码量少却可以实现同样的功能。 接下来就来讲讲kotlin中常见的内联函数,他们在功能上,使用上都大同小异。分别是let函数、with函数、run函数、apply函数、also函数。一个一个来讲解。 从功能返回值上看,let、with、run归为一类,apply和also为一类
173 0
|
20天前
|
移动开发 Java Android开发
构建高效Android应用:探究Kotlin与Java的性能差异
【4月更文挑战第3天】在移动开发领域,性能优化一直是开发者关注的焦点。随着Kotlin的兴起,其在Android开发中的地位逐渐上升,但关于其与Java在性能方面的对比,尚无明确共识。本文通过深入分析并结合实际测试数据,探讨了Kotlin与Java在Android平台上的性能表现,揭示了在不同场景下两者的差异及其对应用性能的潜在影响,为开发者在选择编程语言时提供参考依据。