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/246cb/246cbe9cfb667eef21588fa6950c12f887cc0c8c" alt=""
如图3-4所示,可以查看一下Kotlin Bytecode,会发现它确实是这样的。
data:image/s3,"s3://crabby-images/057b1/057b1f1f0000230948f341b38677c10fd5fe9983" alt=""
图3-4 查看Kotlin函数的字节码
如图3-5所示,再来查看sum(3, 5)的使用,发现它调用了Function2的invoke()方法。
data:image/s3,"s3://crabby-images/83a6c/83a6c0ebf47fffe35a13ca2a2e43297de6af1a81" alt=""
图3-5 sum(3, 5)函数的字节码
换一种工具,如图3-6所示,使用JD-GUI将上述Kotlin代码反编译成Java代码会更加清晰。
data:image/s3,"s3://crabby-images/66dc3/66dc3f0133957b90bc3c0bb7354caa98bbc25270" 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/d7535/d753575806b2470043179ea46d3a1104202455fb" alt=""
如图3-7所示,反编译成Java的代码之后,会发现并没有实现FunctionN接口。
data:image/s3,"s3://crabby-images/da337/da3373245f9e4763082ff9fc625a0b8c2ee34152" alt=""
图3-7 反编译高阶函数
这是因为在Kotlin的filter、forEach函数中都使用了inline,表明这些函数是内联函数。所以,这些函数不需要实例化FunctionN接口,内联函数的特性将会在第4章讲述。
data:image/s3,"s3://crabby-images/9969b/9969bddbf1a92cf092c30dd9a0d86df2a0e00e43" alt=""
Lambda表达式会实现一个FunctionN接口,N的大小由表达式的参数个数决定。在Kotlin 1.3之前不能大于23个,只能是0~22,在Kotlin 1.3之后放宽了限制。
如果Lambda表达式中使用函数作为参数,并且整个高阶函数使用inline修饰,Lambda表达式就不必实现FunctionN接口。