Kotlin 作为一门现代 JVM 语言,提供了出色的 Java 互操作性。为了更好地支持与 Java 代码的交互,Kotlin 提供了一系列 JVM 相关注解。这些注解不仅能帮助我们控制 Kotlin 代码编译成 Java 字节码的行为,还能让我们的 Kotlin 代码更好地被 Java 代码调用。虽然在日常开发中我们最常用的是 @JvmOverloads、@JvmStatic、@JvmName 和 @JvmField
这几个注解,但 Kotlin 其实还提供了更多强大的 JVM 注解。本文系统地整理下这些注解的作用、使用场景和具体示例,便于开发。
ps:以下整理基于kotlin-stdlib-1.7.10.jar!\kotlin\jvm\JvmInline.class
。
为带有默认参数值的函数生成重载方法。
当 Kotlin 函数需要被 Java 代码调用时,特别是函数包含默认参数值的情况。
@JvmOverloads
fun greet(name: String, greeting: String = "Hello") {
println("$greeting, $name!")
}
void greet(String name) {
greet(name, "Hello");
}
void greet(String name, String greeting) {
System.out.println(greeting + ", " + name + "!");
}
生成静态方法或静态属性访问器。
在 companion object 中定义需要作为静态成员的方法或属性。
class MyClass {
companion object {
@JvmStatic
fun staticMethod() { }
@JvmStatic
var staticProperty: String = ""
}
}
指定生成的 Java 类或方法的名称。
@JvmName("filterString")
fun filter(list: List<String>) { }
@JvmName("filterInt")
fun filter(list: List<Int>) { }
指示编译器生成多文件类,将多个文件中的顶级函数和属性合并到一个类中。
需要将分散在多个文件中的相关功能组织在一起时。
// File1.kt
@JvmName("Utils")
@JvmMultifileClass
fun function1() { }
// File2.kt
@JvmName("Utils")
@JvmMultifileClass
fun function2() { }
更改生成的 .class 文件的 JVM 包名。
需要自定义生成的 Java 代码的包名时。
在 Java 字节码中设置 ACC_SYNTHETIC 标志,使目标对 Java 代码不可见。
需要隐藏 Kotlin 特定的目标,使其对 Java 代码不可见,但保持对 Kotlin 代码可见。
@JvmSynthetic
fun internalFunction() { }
指定函数编译为 JVM 方法时应声明的异常。
需要从 Kotlin 代码中抛出 Java 检查异常时。
@Throws(IOException::class)
fun readFile() { }
void readFile() throws IOException { }
指示编译器不要为属性生成 getter/setter,而是将其作为字段暴露。
需要将 Kotlin 属性作为 Java 字段使用时。
class MyClass {
@JvmField
var field: String = ""
}
控制是否生成通配符。
需要控制泛型类型参数的 Java 表示时。
@JvmSuppressWildcards
fun process(list: List<String>) { }
为带声明点变异的类型参数生成通配符。
需要控制泛型类型参数的 Java 表示时。
fun process(@JvmWildcard list: List<String>) { }
指定值类为内联类。
创建零开销的类型安全包装器。
@JvmInline
value class Password(val value: String)
指示编译器将类标记为记录类。
创建不可变的数据类。
@JvmRecord
data class Person(val name: String, val age: Int)
选择合适的注解
性能考虑
代码可维护性
版本兼容性