在 Java 语言 中 , 任何 引用类型变量 都可以为 空 null ; Java 中 八种 基本数据类型 变量 的 默认值 为 0 或 false ;
但是在 Kotlin 语言 中 , 所有的 变量 都是引用类型变量 , 没有基本数据类型 , 默认情况下 所有的变量 都为 非空类型 ;
下面分别定义一个 Java 类 和 Kotlin 脚本 , 在 Kotlin 脚本调用调用 Java 类的成员 ;
代码示例 : 定义一个 Java 函数 , 分别返回 非空字符串 和 空值 ;
public class JavaMethod {// 返回非空字符串public String getName() {return "Tom";}// 返回 nullpublic String getNullName() {return null;}
}
在 Kotlin 中 调用上述类中的两个函数 , 是不会报错的 ;
但是 , 如果调用 空值 的 成员 , 则直接报 空指针异常 ;
代码示例 :
fun main() {val javaMethod = JavaMethod()// 打印两个返回值println(javaMethod.getName())println(javaMethod.getNullName())// 如果调用空值的成员, 则会报错javaMethod.getNullName().length
}
执行结果 :
Tom
null
Exception in thread "main" java.lang.NullPointerExceptionat HelloKt.main(Hello.kt:9)at HelloKt.main(Hello.kt)
在 Kotlin 中 , 凡是 调用 Java 代码 获取的 变量 , 不知道 这个变量 是否为空 , 这种变量的类型 就称为 " 平台类型 " ;
所有的 平台类型 变量 都是 可空的 , Kotlin 会将其自动推断为 可空类型 ;
调用 平台类型 变量 的成员时 , 都必须使用 " ?. " 操作符 进行访问 ;
如下图所示 : 调用 JavaMethod.java 类中的 函数 , 获取的变量 , 被 自动推断为 String? 类型 ;
代码示例 :
fun main() {val javaMethod = JavaMethod()// 打印两个返回值println(javaMethod.getName())println(javaMethod.getNullName())// 调用 Java 函数获取的 平台类型 变量val name = javaMethod.getNullName()println(name?.length)
}
执行结果 :
Tom
null
null
在 Java 中 , 一般使用 @NotNull 和 @Nullable 注解 标记
是否可以为空 ;
Java 代码示例 : 上述代码使用 @NotNull 和 @Nullable 注解 后代码如下 ;
import com.sun.istack.internal.NotNull;
import com.sun.istack.internal.Nullable;public class JavaMethod {@NotNullpublic String getName() {return "Tom";}@Nullablepublic String getNullName() {return null;}
}
在 Kotlin 代码运行时 , 所有的 数据类型都会映射为 Java 类型 ;
代码示例 : 在代码中 , 定义了 Kotlin 中的 Int 类型变量 , 在运行时 , 调用该变量的 .javaClass
查看其映射的 Java 类型 , 最后打印出的结果为 Java 中的 int 类型 ;
fun main() {val number: Int = 1println(number.javaClass)
}
执行结果 :
int
在 Java 中 , 如果要 访问 private 私有属性 , 需要 调用 Getter 和 Setter 方法 ;
在 Kotlin 中 , 直接使用 属性名称 , 即可 访问 Java 中的 private 私有属性 , 该访问包括 读取属性 和 写出属性 操作 ;
代码示例 :
public class JavaMethod {private String name = "Tom";public String getName() {return name;}public void setName(String name) {this.name = name;}
}
var name = javaMethod.name
读取 私有属性 , 调用的是 JavaMethod#getName 函数 ;javaMethod.name = "Jerry"
修改 私有属性 , 调用的是 JavaMethod#setName 函数 ;fun main() {val javaMethod = JavaMethod()var name = javaMethod.nameprintln(name)javaMethod.name = "Jerry"name = javaMethod.nameprintln(name)
}
执行结果 :
Tom
Jerry
在 Java 中调用 Kotlin 脚本中的函数 , 可以直接使用 " Kotlin 文件名 + Kt # 函数名 " 进行调用 , 定义在 Kotlin 文件中的函数相当于 静态函数 , 然后通过静态形式调用 ;
在 Hello.kt 中定义如下函数 : 该函数相当于定义在 HelloKt 类 中的 sayHello 静态函数 ;
fun sayHello() {println("Hello World !")
}
在 Java 代码中调用上述函数 :
public class JavaMethod {public static void main(String[] args) {HelloKt.sayHello();}
}
执行结果 :
Hello World !
分析上述 Kotlin 代码的字节码文件 , 在 Kotlin Bytecode 页面 , 查看其 字节码文件 ;
点击 Decompile 按钮 , 将字节码 反编译回 Java 代码 ,
由下面的代码可知 , 在 Hello.kt 脚本 中 定义 sayHello 函数 , 其对应的 字节码 反编译 后 的 Java 代码 如下 :
import kotlin.Metadata;@Metadata(mv = {1, 4, 2},bv = {1, 0, 3},k = 2,d1 = {"\u0000\b\n\u0000\n\u0002\u0010\u0002\n\u0000\u001a\u0006\u0010\u0000\u001a\u00020\u0001¨\u0006\u0002"},d2 = {"sayHello", "", "KotlinDemo"}
)
public final class HelloKt {public static final void sayHello() {String var0 = "Hello World !";boolean var1 = false;System.out.println(var0);}
}
如果不想 Hello.kt 生成的 Java 类类名为 HelloKt , 可以在 Kotlin 脚本中 使用 @JvmName 注解 修改 Kotlin 生成的 Java 类名 , 相当于 为 Hello.kt 取了一个别名 ;
用法示例 :
@file:JvmName("Hello")
Kotlin 代码示例 :
@file:JvmName("Hello")fun sayHello() {println("Hello World !")
}
Java 代码示例 :
public class JavaMethod {public static void main(String[] args) {Hello.sayHello();}
}
执行结果 :
Hello World !
在快速搜索中 , 选择 Show Kotlin Bytecode 选项 , 查看 Kotlin 的 字节码数据 ;
在 Kotlin Bytecode 界面 , 选择 Decompile 选项 , 将 字节码数据 反编译字节码为 Java 代码 ;
查看生成的 Java 代码 , 可以看到 最终生成的 Java 字节码中 , 类名为 Hello , 使用 @JvmName 注解 成功 修改 Java 编译类名称 ;
import kotlin.Metadata;
import kotlin.jvm.JvmName;@Metadata(mv = {1, 4, 2},bv = {1, 0, 3},k = 2,d1 = {"\u0000\b\n\u0000\n\u0002\u0010\u0002\n\u0000\u001a\u0006\u0010\u0000\u001a\u00020\u0001¨\u0006\u0002"},d2 = {"sayHello", "", "KotlinDemo"}
)
@JvmName(name = "Hello"
)
public final class Hello {public static final void sayHello() {String var0 = "Hello World !";boolean var1 = false;System.out.println(var0);}
}