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