什么是内联函数
这个玩意吸取自C++,如下:
内联函数是C++的增强特性之一,用来降低程序的运行时间。当内联函数收到编译器的指示时,即可发生内联:编译器将使用函数的定义体来替代函数调用语句,这种替代行为发生在编译阶段而非程序运行阶段
举个现实中的例子:
- 通常我要骑车到菜市场买菜,然后再骑车回来,经过这样的折腾,最后我终于吃上肉了。
你看,为了买菜,我花费了大量的时间成本! - ok,现在我在自家的院子种了一些菜,想吃的时候直接拔出来吃就行了。
你看,这样我节省了不少时间啦。
代码中考虑:
- 在kotlin中,函数就是对象,当你调用某个函数的时候,就会创建相关的对象。
你看,这就是空间上的开销! - 当你去调用某个函数的时候,虚拟机会去找到你调用函数的位置。然后执行函数,然后再回到你调用的初始位置。
你看,这就是时间上的开销! - ok,现在,我直接把
调用的函数里面的代码
放到我调用的地方,省去寻找调用函数的位置的时间。
在代码中的例子:
-
正常的函数调用:
-
使用内联:
将fun syaHello()
函数标识为内联inline fun sayHello()
,神奇的事情发生了,你的代码会变成如下的模样:
kotlin中如何定义内联函数呢?
很简单,在函数前加入一个标识:inline
就可以了!
注意:
在kotlin中,函数就是对象,我们可以将函数作为参数传递给函数。
所以问题来了,若我们的内联函数中的参数有函数呢?那这个函数是不是内联函数呢?
当你声明doCal
为内联函数的时候,这个函数所使用的lambda(也就是函数)也默认为内联函数。
当然,我们也可以改变这种行为:
return 在内联函数中的表现
在kotlin中,return 只可以用在有名字的函数,或者匿名函数
中,使得该函数执行完毕。
而针对lambda表达式,你不能直接使用return
你可以使用return+label的形式,将这个lambda结束。
但是
若你的lambda应用在一个内联函数的时候,这时候你可以在lambda中使用return
可以这么理解,内联函数在编译的时候,将相关的代码贴入你调用的地方。
lambda表达式就是一段代码而已,这时候你在lambda中的return,相当于在你调用的方法内return
这么个玩法,可以使用在什么地方?
return直接作用在你使用的地方
举个例子:
我们通常会对数组进行操作,如下
就像执行了你写的代码一样,该返回就返回了,最后代码不会执行
println("end")
。
内联属性
对属性来说,我们会有get,set的方法来操作这个属性。
get,set就是个函数,我们可以标识他们为内联函数:
val foo: Foo
inline get() = Foo()
var bar: Bar
get() = ...
inline set(v) { ... }
当然,你也可以这么写:
inline var bar: Bar
get() = ...
set(v) { ... }
其他
指定参数的类型
有这么个例子:
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>()
那么该怎么写?
内联函数支持 具体的类型参数的声明 reified
这么写的:
inline fun <reified T> TreeNode.findParentOfType(): T? {
var p = parent
while (p != null && p !is T) {
p = p.parent
}
return p as T?
}