kotlin-inline函数

本文探讨了高阶函数在运行时的效率损失及其解决办法——内联技术。介绍了内联如何通过避免函数对象的创建和虚拟调用来提高性能,特别是在循环中。同时,展示了内联在参数类型优雅写法和属性访问上的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

高阶函数是将函数用作参数或返回值的函数。如lambda表达式

1.高阶函数缺点: 使用高阶函数会带来一些运行时的效率损失:每一个函数都是一个对象,并且会捕获一个闭包。 即那些在函数体内会访问到的变量。 内存分配(对于函数对象和类)和虚拟调用会引入运行时间开销。

尤其在循环中,能节省很多开销。

于是,inline诞生:

lock(l) { foo() }
复制代码
注:上面的代码,编译器会自动转成,不会生成函数对象
l.lock()
try {
    foo()
}
finally {
    l.unlock()
}
复制代码

具体做法:

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

复制代码

2.优雅的参数类型写法

如:有时候我们需要访问一个作为参数传给我们的一个类型:

fun TreeNode.findParentOfType(clazz: Class): 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() 为能够这么做,内联函数支持具体化的类型参数,于是我们可以这样写:

inline fun TreeNode.findParentOfType(): T? { var p = parent while (p != null && p !is T) { p = p.parent } return p as T? } 我们使用 reified 修饰符来限定类型参数,现在可以在函数内部访问它了, 几乎就像是一个普通的类一样。由于函数是内联的,不需要反射,正常的操作符如 !is 和 as 现在都能用了。此外,我们还可以按照上面提到的方式调用它:myTree.findParentOfType()。

3.内联属性

val foo: Foo
    inline get() = Foo()
​
var bar: Bar
    get() = ……
    inline set(v) { …… }
你也可以标注整个属性,将它的两个访问器都标记为内联:

inline var bar: Bar
    get() = ……
    set(v) { …… }
在调用处,内联访问器如同内联函数一样内联
复制代码

转载于:https://juejin.im/post/5cadacc96fb9a068b2296827

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值