我們知道kotlin可以為方法參數(shù)設(shè)置默認(rèn)值。見下圖
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兩個(gè)方法,testMethod方法 param1參數(shù)默認(rèn)值為Hello,調(diào)用時(shí)如果不傳param1的值,則必須指明后面參數(shù)的方法名稱的值。如果方法中最后一個(gè)參數(shù)含有默認(rèn)值,調(diào)用時(shí)可以無需指名要傳參數(shù)的名稱,如testMethod1的方法調(diào)用。
那方法默認(rèn)參數(shù)到底是怎樣實(shí)現(xiàn)的,轉(zhuǎn)換成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);
}
原來編譯時(shí)把當(dāng)前類下有默認(rèn)參數(shù)的方法生成一個(gè)靜態(tài)方法,方法名字為原方法名稱后面加了$default后綴,我們以testMethod來看,會(huì)生成testMethod default方法,其中的參數(shù)第一個(gè)var0為當(dāng)前類的實(shí)例,var1為param1,var2為param2,
var3與有默認(rèn)參數(shù)在靜態(tài)方法參數(shù)的index做&操作,不等于表示參數(shù)沒有傳值,使用默認(rèn)參數(shù)。如果為0表示有傳值直接調(diào)用實(shí)例的具體方法實(shí)現(xiàn)。
其實(shí)kotlin的類擴(kuò)展方法與擴(kuò)展屬性都是在編譯時(shí)期生成一些靜態(tài)的類方法,
而用Object定義單例就是在編譯進(jìn)生成靜態(tài)代碼塊初始化實(shí)例。反編譯成java代碼后一目了然。