Kotlin-45.Java调用kotlin之三(Call Kotlin from Java)

官方文档: http://kotlinlang.org/docs/reference/java-to-kotlin-interop.html

8.@JvmName解决java方法签名相同(Handling signature clashes)

最突出的例子是由于类型擦除(type erasure)引发:
    // 类型擦除: 无法区分List<String>和List<Int>
    fun List<String>.filterValid(): List<String>
    fun List<Int>.filterValid(): List<Int>
    这两个函数在java中不能同时定义,因为它们的JVM签名相同: filterValid(Ljava/util/List;)Ljava/util/List

在Kotlin中用相同名称,需要用@JvmName标注其中的一个(或两个),并指定不同名称:
    fun List<String>.filterValid(): List<String>

    @JvmName("filterValidInt")
    fun List<Int>.filterValid(): List<Int>

    在Kotlin中,可以用相同名称filterValid()访问;
    在Java中,需要分别用filterValid()和filterValidInt()访问

同样注解@JvmName也适用于属性x和函数getX():
    val x: Int
        @JvmName("getX_prop")
        get() = 15

    fun getX() = 10

    在Java中,需要分别用getX_prop()和getX()访问

9.Java方法重载(Overloads Generation)

如果Kotlin函数的参数有默认值并且使用@JvmOverloads注解,
那么在Java中多个重载方法, 示例如下:
    @JvmOverloads fun f(a: String, b: Int = 0, c: String = "abc") {
    }

    // kotlin函数的每一个有默认值的参数,都会重载生成一个额外java方法:    
    void f(String a, int b, String c) {            
    }
    void f(String a, int b) {            
    }
    void f(String a) {            
    }

该注解也适用于构造函数和静态方法, 但不能用于抽象方法(包括接口的方法)

对于次构造函数(Secondary Constructors), 如果所有参数都有默认值,
那么会生成一个公有public的无参构造函数(没有@JvmOverloads注解也会生效)!

10.受检异常(Checked Exception)

从前几章《kotlin-33.异常(Exception)》可知,Kotlin没有受检异常!
所以Kotlin函数签名不会声明抛出异常(throws Exception),例如:    
    // kotlin (example.kt), 抛出异常
    package demo
    fun foo() {
        throw IOException()
    }

    // kotlin编译生成的Java方法
    public void foo() { // 错误: foo()没有声明throws IOException
        throw IOException()
    }

因为由kotlin函数生成的Java方法foo()没有声明throws IOException, 所以Java编译器报错!
为了解决这个问题,需要在Kotlin中使用@Throws注解(相当于在Java中声明throws IOException)
    // kotlin
    @Throws(IOException::class)
    fun foo() {
        throw IOException()
    }

    // kotlin编译生成的Java方法
    public void foo() throws IOException {
        throw IOException()
    }

11.型变泛型(Variant generics)

当Kotlin类使用声明处型变(declaration-site variance)时,在Java中有两种用法!
示例:
    // kotlin
    class Box<out T>(val value: T)
    interface Base
    class Derived : Base

    fun boxDerived(value: Derived): Box<Derived>{            
    }
    fun unboxBase(box: Box<Base>): Base {            
    }
    unboxBase(boxDerived("s")) // 正确: 在Java中Box<Base>泛型是不型变的!

    // 将上述kotlin函数转换成Java方法       
    Box<Derived> boxDerived(Derived value) {            
    }
    Base unboxBase(Box<Base> box) {            
    }
    unboxBase(boxDerived("s")) // 错误: 因为在Java中Box<Base>泛型是不型变的!

    //使用Java通配符类型<? extends Base>模拟kotlin声明处型变
    Base unboxBase(Box<? extends Base> box) {            
    }
    unboxBase(boxDerived("s")) // 正确

此外, 对于协变定义的Box(在kotlin中Box<in T>), 在java中使用Box<? extends Super>
注意:当参数类型是final时,通配符? extends没有意义,
        例如在Box<String>中的String类是final,没有子类(不能被继承extends)

如果在默认没有通配符处要求java泛型通配符, 可以使用@JvmWildcard注解:
    // kotlin函数
    fun boxDerived(value: Derived): Box<@JvmWildcard Derived> {            
    }

    // 转换成java方法 
    Box<? extends Derived> boxDerived(Derived value) {        
    }

相反,如果不需要泛型通配符,可以使用@JvmSuppressWildcards注解;
@JvmSuppressWildcards不仅可用于单个类型参数,还可用于整个声明(如函数或类),从而抑制其中的所有通配符!
    // kotlin函数
    fun unboxBase(box: Box<@JvmSuppressWildcards Base>): Base {            
    }

    // 转换成java方法 
    Base unboxBase(Box<Base> box) {            
    }

12.Nothing类型翻译(Translation of type Nothing)

类型Nothing是Kotlin特有的,在Java中没有对应类型!
每个Java引用类型(包括java.lang.Void)都接受null, 但是kotlin的Nothing不行!
Nothing类型不能在Java世界中准确表示,所以Nothing类型在java中会消失(原始类型raw type):
    // kotlin的Nothing类型
    fun emptyList(): List<Nothing> = listOf()
    
    // 翻译成转换成java方法, List<Nothing>变成原始类型List, Nothing类型消失了
    List emptyList() {            
    }

简书:http://www.jianshu.com/p/acdd8ba0830b
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/75452680
GitHub博客:http://lioil.win/2017/07/19/Kotlin-kotlinInJava3.html
Coding博客:http://c.lioil.win/2017/07/19/Kotlin-kotlinInJava3.html

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

推荐阅读更多精彩内容