Kotlin:你必须要知道的-inline-noinline-crossinline

}

public static final int sum(int a, int b) {
return a + b;
}

sum 函数的实现代码被直接拷贝到了调用的地方。

上面两个使用实例并没有体现出 inline 的优势。当你的函数中有 lambda 形参时,inline 的优势才会体现。

###inline function with lambda parameters

考虑如下代码,会被编译成怎样的 Java 代码?

fun sum(a: Int, b: Int, lambda: (result: Int) -> Unit): Int {
val r = a + b
lambda.invoke®
return r
}

fun main(args: Array) {
sum(1, 2) { println(“Result is: $it”) }
}

反编译为 Java:

public static final int sum(int a, int b, @NotNull Function1 lambda) {
//…
int r = a + b;
lambda.invoke®;
return r;
}

public static final void main(@NotNull String[] args) {
//…
sum(1, 2, (Function1)null.INSTANCE);
}

(Function1)null.INSTANCE,是由于反编译器工具在找不到等效的 Java 类时的显示的结果。
我传递的那个 lambda 被转换为 Function1 类型,它是 Kotlin 函数(kotlin.jvm.functions包)的一部分,它以 1 结尾是因为我们在 lambda 函数中传递了一个参数(result:Int)。

再考虑如下代码:

fun main(args: Array) {
for (i in 0…10) {
sum(1, 2) { println(“Result is: $it”) }
}
}

我在循环中调用 sum 函数,每次传递一个 lambda 打印结果。反编译为 Java:

for(byte var2 = 10; var1 <= var2; ++var1) {
sum(1, 2, (Function1)null.INSTANCE);
}

可见在每次循环里都会创建一个 Function1 的实例对象。这里就是性能的优化点所在,如何避免在循环里创建新的对象?

1.在循环外部创建 lambda 对象

val l: (r: Int) -> Unit = { println(it) }

for (i in 0…10) {
sum(1, 2, l)
}

反编译为 Java:

Function1 l = (Function1)null.INSTANCE;
int var2 = 0;

for(byte var3 = 10; var2 <= var3; ++var2) {
sum(1, 2, l);
}

只会创建一个 Function 对象

2.使用 inline:

fun main(args: Array) {
for (i in 0…10) {
sum(1, 2) { println(“Result is: $it”) }
}
}

inline fun sum(a: Int, b: Int, lambda: (result: Int) -> Unit): Int {
val r = a + b
lambda.invoke®
return r
}

反编译为 Java:

public static final void main(@NotNull String[] args) {
//…
int var1 = 0;

for(byte var2 = 10; var1 <= var2; ++var1) {
byte a i v = 1 ; i n t b iv = 1; int b iv=1;intbiv = 2;
int r i v = a iv = a iv=aiv + b i v ; S t r i n g v a r 9 = " R e s u l t i s : " + r iv; String var9 = "Result is: " + r iv;Stringvar9="Resultis:"+riv;
System.out.println(var9);
}
}

lambda 代码在编译时被拷贝到调用的地方, 避免了创建 Function 对象。

###inline 注意事项

#####public inline 函数不能访问私有属性

class Demo(private val title: String) {

inline fun test(l: () -> Unit) {
println(“Title: $title”) // 编译错误: Public-Api inline function cannot access non-Public-Api prive final val title
}

// 私有的没问题
private inline fun test(l: () -> Unit) {
println(“Title: $title”)
}
}

#####注意程序控制流

当使用 inline 时,如果传递给 inline 函数的 lambda,有 return 语句,那么会导致闭包的调用者也返回。

例子:

inline fun sum(a: Int, b: Int, lambda: (result: Int) -> Unit): Int {
val r = a + b
lambda.invoke®
return r
}

fun main(args: Array) {
println(“Start”)
sum(1, 2) {
println(“Result is: $it”)
return // 这个会导致 main 函数 return
}
println(“Done”)
}

反编译 Java:

public static final void main(@NotNull String[] args) {
String var1 = “Start”;
System.out.println(var1);
byte a i v = 1 ; i n t b iv = 1; int b iv=1;intbiv = 2;
int r i v = a iv = a iv=aiv + b i v ; S t r i n g v a r 7 = " R e s u l t i s : " + r iv; String var7 = "Result is: " + r iv;Stringvar7="Resultis:"+riv;
System.out.println(var7);
}

反编译之后也能看到,lambda return 之后的代码不会执行。

如何避免?

可以使用 return@label 语法,返回到 lambda 被调用的地方。

fun main(args: Array) {
println(“Start”)
sum(1, 2) {
println(“Result is: $it”)
return@sum
}
println(“Done”)
}

###noinline

当一个 inline 函数中,有多个 lambda 作为参数时,可以在不想内联的 lambda 前使用 noinline 声明.

inline fun sum(a: Int, b: Int, lambda: (result: Int) -> Unit, noinline lambda2: (result: Int) -> Unit): Int {
val r = a + b
lambda.invoke®
lambda2.invoke®
return r
}

fun main(args: Array) {
sum(1, 2,
{ println(“Result is: $it”) },
{ println(“Invoke lambda2: $it”) }
)
}

反编译 Java:

public static final int sum(int a, int b, @NotNull Function1 lambda, @NotNull Function1 lambda2) {
int r = a + b;
lambda.invoke®;
lambda2.invoke®;
return r;
}

public static final void main(@NotNull String[] args) {
byte a i v = 1 ; b y t e b iv = 1; byte b iv=1;bytebiv = 2;
Function1 lambda2 i v = ( F u n c t i o n 1 ) n u l l . I N S T A N C E ; i n t r iv = (Function1)null.INSTANCE; int r iv=(Function1)null.INSTANCE;intriv = a i v + b iv + b iv+biv;
String var8 = "Result is: " + r i v ; S y s t e m . o u t . p r i n t l n ( v a r 8 ) ; l a m b d a 2 iv; System.out.println(var8); lambda2 iv;System.out.println(var8);lambda2iv.invoke(r$iv);
}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的相关的几十套腾讯、头条、阿里、美团等公司21年的面试专题,其中把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分免费分享给大家,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家~

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

【延伸Android必备知识点】

这里只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img
部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-Rez76mw5-1712857680348)]

  • 18
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值