data:image/s3,"s3://crabby-images/34bcb/34bcb216b9b7d33d0e22778444db7d94d20c7532" alt="Kotlin进阶实战"
3.2.7 换个角度看Lambda表达式
Kotlin的代码最终还是会编译成.class文件,由JVM进行加载。
1.示例1
我们曾经讲过,sum这个函数本质上实现了kotlin.jvm.functions.Function2接口,即:
data:image/s3,"s3://crabby-images/86798/86798c8eb140492c09c2ea3c8c096ed15715f72e" alt=""
如图3-4所示,可以查看一下Kotlin Bytecode,会发现它确实是这样的。
data:image/s3,"s3://crabby-images/a2b1c/a2b1c4c4e7922a9d4d8ed7848bfa653146011cd3" alt=""
图3-4 查看Kotlin函数的字节码
如图3-5所示,再来查看sum(3, 5)的使用,发现它调用了Function2的invoke()方法。
data:image/s3,"s3://crabby-images/19aae/19aae3942335e9d9bd50088096fea878c4a81a8d" alt=""
图3-5 sum(3, 5)函数的字节码
换一种工具,如图3-6所示,使用JD-GUI将上述Kotlin代码反编译成Java代码会更加清晰。
data:image/s3,"s3://crabby-images/c70bb/c70bb7325e1f6ecb396e103acbc22aa0342bf692" alt=""
图3-6 使用JD-GUI进行反编译
有多种方式可以反编译成字节码,例如使用IDEA自带的工具Tools->Kotlin->Show Kotlin Bytecode,或者使用javap命令反编译.class文件,亦或者使用BytecodeViewer反编译.class文件,都能够将Kotlin的代码反编译成字节码。有时字节码看不清晰,反编译成Java代码效果会更好。
2.示例2
下面是使用filter、forEach的例子,它们都是高阶函数。
data:image/s3,"s3://crabby-images/1b334/1b334b4c9030c3ddd595f04358e4a354ee7587cd" alt=""
如图3-7所示,反编译成Java的代码之后,会发现并没有实现FunctionN接口。
data:image/s3,"s3://crabby-images/b2226/b222621b4705dd336bd764d0eff647c4f880d524" alt=""
图3-7 反编译高阶函数
这是因为在Kotlin的filter、forEach函数中都使用了inline,表明这些函数是内联函数。所以,这些函数不需要实例化FunctionN接口,内联函数的特性将会在第4章讲述。
data:image/s3,"s3://crabby-images/5d182/5d18204ecfb3f145c4f59181c554757dd2fc7a11" alt=""
Lambda表达式会实现一个FunctionN接口,N的大小由表达式的参数个数决定。在Kotlin 1.3之前不能大于23个,只能是0~22,在Kotlin 1.3之后放宽了限制。
如果Lambda表达式中使用函数作为参数,并且整个高阶函数使用inline修饰,Lambda表达式就不必实现FunctionN接口。