从Java调用Kotlin代码

在平时开发中不可避免的需要从Java调用Kotlin,今天就学习一下哪些是Java调用Kotlin更加方便的注解。

@JvmStatic

object StringUtil {
    init {
        println()
    }
    
    fun changeTo(value: String) {

    }
}

StringUtil.INSTANCE.changeTo(data);

在Java中调用对象声明中的方法时,需要带INSTANCE,不像调用Java静态方法,直接使用类名+方法名,因为Kotlin反编译生成的Java代码就有一个静态INSTANCE实例

public final class StringUtil {
   @NotNull
   public static final StringUtil INSTANCE;

   public final void changeTo(@NotNull String value) {
      Intrinsics.checkNotNullParameter(value, "value");
   }

   private StringUtil() {
   }

   static {
      StringUtil var0 = new StringUtil();
      INSTANCE = var0;
      System.out.println();
   }
}

通过这个实例,可以直接调用类中的方法,如果我们想像调用Java方法那样调用Kotlin 对象声明的方法,应该怎么办呢,此时就需要用到@JvmStatic注解。

object StringUtil {
    init {
        println()
    }
    
    @JvmStatic
    fun changeTo(value: String) {

    }
}

StringUtil.changeTo(data);

添加@JvmStatic之后可以通过类名直接调用方法,反编译之后的Java代码是什么样呢?

public final class StringUtil {
   @NotNull
   public static final StringUtil INSTANCE;

   @JvmStatic
   public static final void changeTo(@NotNull String value) {
      Intrinsics.checkNotNullParameter(value, "value");
   }

   private StringUtil() {
   }

   static {
      StringUtil var0 = new StringUtil();
      INSTANCE = var0;
      System.out.println();
   }
}

changeTo方法变成了静态方法,由此,我们可知@JvmStatic的作用。
@JvmStatic可以用来修饰方法或属性,声明为所注释的方法生成对应的静态方法,为所注释的属性生成静态的getter/setter。

@JvmName

@JvmName可以指定Kotlin生成的字节码中类,属性和方法的命名方式。
我们创建一个Kotlin文件StringUtil.kt,一个顶层函数


package com.lkk.kotlinbasic.myapp

fun changeTo(name: String) {
    println(name)
}

StringUtilKt.changeTo("hello");

默认生成的类名为StringUtilKt,反编译后的Java代码如下:

public final class StringUtilKt {
   public static final void changeTo(@NotNull String name) {
      Intrinsics.checkNotNullParameter(name, "name");
      System.out.println(name);
   }
}

接下来我们使用@JvmName指定生成的类名

@file:JvmName("StringUtil")
package com.lkk.kotlinbasic.myapp

fun changeTo(name: String) {
    println(name)
}

反编译之后的Java代码如下

public final class StringUtil {
   public static final void changeTo(@NotNull String name) {
      Intrinsics.checkNotNullParameter(name, "name");
      System.out.println(name);
   }
}

生成的类名正是咱们指定的类名。
@JvmName同样可以用来注解属性

class User {
    val hasSystemAccess
        get() = ""
}

我们在Java中调用时只能使用user.getHasSystemAccess(),如果我们想修改自动生成get方法的名称,可以使用@JvmName,这里有两种写法

class User {
    val hasSystemAccess
        @JvmName("hasSystemAccess")
        get() = ""
}

class User {
    @get:JvmName("hasSystemAccess")
    val hasSystemAccess
        get() = ""
}

这两种方式都能实现咱们得目的,参考最终生成的Java代码

public final class User {
   @JvmName(
      name = "hasSystemAccess"
   )
   @NotNull
   public final String hasSystemAccess() {
      return "";
   }
}

通过使用 @JvmName 注解,Kotlin 会为带注解的项生成具有指定名称(而非默认名称)的字节码,同样可以使用@set:JvmName来修改setter的名称,此处不再赘述。

@JvmOverloads

Kotlin支持参数默认值,但Java不支持,从Java调用Kotlin带默认值的构造方法或函数时需要使用@JvmOverloads注解。

class User(name: String = "hello") {
    @JvmOverloads
    fun printUser(name: String, age: Int = 18) {
        print("$name, $age")
    }
}
   @JvmOverloads
   public final void printUser(@NotNull String name) {
      printUser$default(this, name, 0, 2, (Object)null);
   }

   @JvmOverloads
   public final void printUser(@NotNull String name, int age) {
      Intrinsics.checkNotNullParameter(name, "name");
      String var3 = name + ", " + age;
      System.out.print(var3);
   }

给方法printUser添加@JvmOverloads之后,生成了这个方法的重载版本。现在在Java代码中既可以调用printUser(@NotNull String name) ,也可以调用printUser(@NotNull String name, int age),没提交@JvmOverloads,只能调用printUser(@NotNull String name, int age)。
我试了一下给构造方法添加@JvmOverloads,添加之前和添加之后好像没有区别。

@JvmField

@JvmField用来注解属性,添加注解之后Kotlin不会为其自动生成getter/setter,并把它作为一个属性暴露。

class User @JvmOverloads constructor(@JvmField val id: Int, name: String = "hello") {
    @JvmOverloads
    fun printUser(name: String, age: Int = 18) {
        print("$name, $age")
    }
}

我们为id属性添加了@JvmField注解,可以看到反编译之后的Java代码并没有对应的getter/setter,并且在Java代码中可以直接调用id,而不需要使用getter方法。
对于object中的常量,还是建议使用const。

@Throws

Java 中有“受检异常”的概念,但Kotlin 没有受检异常,这就会导致在Java中调用Kotlin抛出受检异常的函数时报错,此时需要抛出@Throws,指示 Java 代码 Kotlin 函数会抛出异常。

fun openFile(filePath: String) {
        val outputFile = File(filePath)
        if (!outputFile.canWrite()) {
            throw FileNotFoundException("Could not write to file: $filePath")
        }
}

user.openFile("");

创建一个抛出异常的方法如上,并在Java侧调用此方法,代码编译时会抛出异常
java.io.FileNotFoundException: Could not write to file:

public final void openFile(@NotNull String filePath) {
      Intrinsics.checkNotNullParameter(filePath, "filePath");
      File outputFile = new File(filePath);
      if (!outputFile.canWrite()) {
         throw (Throwable)(new FileNotFoundException("Could not write to file: " + filePath));
      }
   }

这是编译之后Java代码。

@Throws(IOException::class)
    fun openFile(filePath: String) {
        val outputFile = File(filePath)
        if (!outputFile.canWrite()) {
            throw FileNotFoundException("Could not write to file: $filePath")
        }
    }

我们给openFile添加@Throws(IOException::class)注解,此时看一下反编译之后的Java代码

public final void openFile(@NotNull String filePath) throws IOException {
      Intrinsics.checkNotNullParameter(filePath, "filePath");
      File outputFile = new File(filePath);
      if (!outputFile.canWrite()) {
         throw (Throwable)(new FileNotFoundException("Could not write to file: " + filePath));
      }
   }


try {
            user.openFile("");
        } catch (IOException e) {
            throw new RuntimeException(e);
 }

在函数上添加了throws,此时在Java中再调用此方法,需要try-catch或继续往上抛异常。

Over

参考

  1. 从 Java 调用 Kotlin 代码
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,907评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,987评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,298评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,586评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,633评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,488评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,275评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,176评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,619评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,819评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,932评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,655评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,265评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,871评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,994评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,095评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,884评论 2 354

推荐阅读更多精彩内容