我们知道kotlin可以为方法参数设置默认值。见下图
class TestObject {
companion object {
fun test() {
val testObject = TestObject()
testObject.testMethod(param2 = "world")
testObject.testMethod1("so")
}
}
fun testMethod1(param1: String,param2: String="what") {
println("$param1 $param2")
}
fun testMethod(param1:String="Hello",param2: String) {
println("$param1 $param2")
}
}
我们定义了testMethod与testMethod1两个方法,testMethod方法 param1参数默认值为Hello,调用时如果不传param1的值,则必须指明后面参数的方法名称的值。如果方法中最后一个参数含有默认值,调用时可以无需指名要传参数的名称,如testMethod1的方法调用。
那方法默认参数到底是怎样实现的,转换成Java文件后就一目了然。如下:
public final class TestObject {
@NotNull
public static final TestObject.Companion Companion = new TestObject.Companion((DefaultConstructorMarker)null);
public final void testMethod1(@NotNull String param1, @NotNull String param2) {
Intrinsics.checkParameterIsNotNull(param1, "param1");
Intrinsics.checkParameterIsNotNull(param2, "param2");
String var3 = param1 + ' ' + param2;
boolean var4 = false;
System.out.println(var3);
}
// $FF: synthetic method
public static void testMethod1$default(TestObject var0, String var1, String var2, int var3, Object var4) {
if ((var3 & 2) != 0) {
var2 = "what";
}
var0.testMethod1(var1, var2);
}
public final void testMethod(@NotNull String param1, @NotNull String param2) {
Intrinsics.checkParameterIsNotNull(param1, "param1");
Intrinsics.checkParameterIsNotNull(param2, "param2");
String var3 = param1 + ' ' + param2;
boolean var4 = false;
System.out.println(var3);
}
// $FF: synthetic method
public static void testMethod$default(TestObject var0, String var1, String var2, int var3, Object var4) {
if ((var3 & 1) != 0) {
var1 = "Hello";
}
var0.testMethod(var1, var2);
}
原来编译时把当前类下有默认参数的方法生成一个静态方法,方法名字为原方法名称后面加了$default后缀,我们以testMethod来看,会生成testMethod default方法,其中的参数第一个var0为当前类的实例,var1为param1,var2为param2,
var3与有默认参数在静态方法参数的index做&操作,不等于表示参数没有传值,使用默认参数。如果为0表示有传值直接调用实例的具体方法实现。
其实kotlin的类扩展方法与扩展属性都是在编译时期生成一些静态的类方法,
而用Object定义单例就是在编译进生成静态代码块初始化实例。反编译成java代码后一目了然。