1、@JvmOverloads
为了解决 Java 不能重载 kotlin 有默认参数的方法
Kotlin中代码:
fun show(name: String, age: Int = 20) {
println("name:$name, age:$age")
}
fun main() {
show("王五")
}
Java 中调用:
所以需要在 Kotlin 方法上添加 @JvmOverloads:
public class Test {
public static void main(String[] args) {
PersonFileKt.show("李四");
}
}
自定义View的构造方法中:
class CustomView @JvmOverloads constructor(
private val context: Context,
attrs: AttributeSet? = null,
defStyleAttr: Int = 0
) : View(mContext, attrs, defStyleAttr)
就相当于写了三个构造方法:
public class CustomView extends View{
public CustomView (Context context) {
this(context, null);
}
public CustomView (Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public CustomView (Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
}
2、@JvmField
可以给 Kotlin 属性添加 @JvmField
注解,暴露它的支持字段给 Java 调用者,从而避免使用 getter 方法。
// Kotlin
class Spellbook {
@JvmField
val spells = listOf("Magic Ms. L", "Lay on Hans")
}
// Java
Spellbook spellbook = new Spellbook();
for (String spell : spellbook.spells) {
System.out.println(spell);
}
@JvmField
注解还能用来以静态方式提供伴生对象里定义的值。
// Kotlin
class Spellbook {
...
companion object {
val MAX_SPELL_COUNT = 10
}
}
// Java
Spellbook.Companion.getMAX_SPELL_COUNT();
// Kotlin
class Spellbook {
...
companion object {
@JvmField
val MAX_SPELL_COUNT = 10
}
}
// Java
System.out.println(Spellbook.MAX_SPELL_COUNT);
3、@JvmStatic
@JvmStatic
注解的作用类似于 @JvmField
,允许直接调用伴生对象里的函数。
// Kotlin
class Spellbook {
@JvmField
val spells = listOf("Magic Ms. L", "Lay on Hans")
companion object {
@JvmField
val MAX_SPELL_COUNT = 10
@JvmStatic
fun getSpellbookGreeting() = println("I am the Great Grimoire!")
}
}
// Java
Spellbook.getSpellbookGreeting();
4、@JvmName
为了解决 Java 调用 Kotlin 文件中的字段或者方法,需要在原来 Kotlin 文件名基础后面加 Kt,加上 @JvmName 就可以去掉后面的 Kt,直接文件名调用了。
@file:JvmName("Person")
比如如下Kotlin 文件:
// 文件名 PersonFileKt
package annotations
fun getNameValueInfo(str: String) = println(str)
fun main() {
}
/*
背后生成的代码:
public final class PersonFileKt {
public static final void getNameValueInfo(@NotNull String str) {
System.out.println(str);
}
public static final void main() {
}
// $FF: synthetic method
public static void main(String[] var0) {
main();
}
}
*/
java中调用:
public class Test {
public static void main(String[] args) {
PersonFileKt.getNameValueInfo("张三");
}
}
@file:JvmName("Person")
必须写在包名的前面,否则无法起作用
@file:JvmName("Person")
package annotations
fun getNameValueInfo(str: String) = println(str)
fun main() {
}
/*
背后生成的代码:
public final class PersonFileKt {
public static final void getNameValueInfo(@NotNull String str) {
System.out.println(str);
}
public static final void main() {
}
// $FF: synthetic method
public static void main(String[] var0) {
main();
}
}
*/
Java 中调用,就可以直接使用文件名 .
调用了:
public class Test {
public static void main(String[] args) {
Person.getNameValueInfo("张三");
}
}
5、@JvmMultifileClass
如果我们在同一包名下面的两个 kt 文件中,用 @file:JvmName("Person")
注解了一样的 class 名字,就会报错,需要添加 @JvmMultifileClass
解决
// PersonFile.kt
@file:JvmName("Person")
@file:JvmMultifileClass
package annotations
fun getNameValueInfo(str: String) = println(str)
// PersonFile2.kt
@file:JvmName("Person")
@file:JvmMultifileClass
package annotations
fun getNameValueInfo2(str: String) = println(str)
Java 中调用:
public class Test {
public static void main(String[] args) {
Person.getNameValueInfo("张三");
Person.getNameValueInfo2("李四");
}
}