Kotlin和Java混合使用确实是体验不佳的。没有写纯纯的Kotlin那么爽的。Kotlin 在设计之初就考虑了与 Java 的互操作性。我们可以从 Kotlin 中自然地调用现存的 Java 代码。但是,如果在Kotlin中调用Java代码中的丑陋的特性(比如说,null),这确实是不得已的事情。怎么调用来自Java中的null?
Java 中的任何引用都可能是null,这样我们在使用 Kotlin调用来自 Java 的对象的时候就有可能会出现空安全的问题。Java 声明的类型在 Kotlin 中会被特别对待并称为平台类型(platform types )。对这种类型的空检查会放宽,因此它们的安全保证与在 Java 中相同。
请看以下示例:
@RunWith(JUnit4::class)
class CallingJavaNullSafe {
@Test fun testCallingJavaNullSafe() {
val product = Product()
// product.name = null
product.category = "金融财务类"
product.gmtCreated = Date()
product.gmtModified = Date()
println(JSONUtils.toJsonString(product)) val name = product.name
println("product name is ${name}")
val eqName = name == "账务系统"
println(eqName)
name.substring(1)
}
}
上面的代码可以正确编译通过。Kotlin编译器对来自Java的空值name(平台类型)放宽了空检查name.substring(1)。但是这样的空指针异常仍然会在运行时抛出来。
运行上面的代码,我们可以看到输出:
{"category":"金融财务类","gmtCreated":1500050426817,"gmtModified":1500050426817}
product name is null
falsenull cannot be cast to non-null type java.lang.String
kotlin.TypeCastException: null cannot be cast to non-null type java.lang.String
at com.easy.kotlin.CallingJavaNullSafe.testCallingJavaNullSafe(CallingJavaNullSafe.kt:27)
我们没有设置name的值,在Java它就是null。我们在Kotlin代码中使用了这个name进行计算,我们可以看出:
val eqName = name == "账务系统"
println(eqName)
可以正确输出false。这表明Kotlin的判断字符串是否相等已经对null的情况作了判断处理,这样的代码如果在Java中调用 name.equals("账务系统") 就该抛空指针异常了。
但是当我们直接使用name这个值来调用name.substring(1)的时候,Kotlin编译器不会检查这个空异常,但是运行时还是要报错的:null cannot be cast to non-null type java.lang.String。
如果我们不想看到这样的异常,而是当name是null的时候,安静的输出null,直接使用Kotlin中的空安全的调用 .? :
name?.substring(1)
这样,运行的时候不会抛出异常,直接安静的返回null。
平台类型不能在程序中显式表述,因此在语言中没有相应语法。 然而,编译器和 IDE 有时需要(在错误信息中、参数信息中等)显示他们,所以我们用一个助记符来表示他们:
T! : 表示 T 或者 T?
(Mutable) Collection! : 表示 “可以可变或不可变、可空或不可空的 T 的 Java 集合”
Array<(out) T>! : 表示可空或者不可空的 T(或 T 的子类型)的 Java 数组不知道这么说你明白不