目录
1、对null的约定
kotlin对变量的null进行了区分
var name: String = "Zhang San" //为不可null的变量
var name1: String? = null //为可null的变量
var name2: String = null //报错
相同数据类型的可null和不可null的两个变量是两种完全不同的类型的变量
- 1、它们之间不能直接互相赋值
- 2、不可null和可null的两种类型之间的转换类似于int和float (可null是不可null的高级类型)
private fun testNull(){
name=name1 //1、报错,不可null的变量不能直接转换为可null
name=name1!! //2、可以加两个感叹号进行类型强转
name1=name //3、不可null可的变量可以隐式转换为可null
}
2、Java中对null值处理
java中对null的处理是给了我们两个注解:
@NotNull 被注解的变量不能赋值为null,当赋值为null的时候会给出黄色的警告 @Nullable 被注解的变量可以注意:这些注解只会在我们书写代码的时候,给我们一个黄色警告,来让我们避免的错误的赋值为null,但是当我们不小心传了一个null的时候他们在运行的时候是会报错的,但是在Kotlin,你如果声明该类型为可null,则不会报错。
左边是我们的源码,右边是编译后的class文件,明显看出,这两个注解只是警告作用,并没有实际的判null操作。
我们看到上面代码中,我们对变量name2和printNam()函数的参数都用@NotNull 进行了注解,在我们给它们赋值为null的时候,会给我们一个黄色的警告,并且会给我们说明。但是我们强行运行这段代码的时候,它是会直接报错的。
而用了@Nullable 注解的pringName1()函数的参数时,它会警告我们让我们对str进行if null值的判断。
3、Kotlin中 null安全的实现
null安全的实现其实就是对你声明的变量进行两步null判断处理,下面有源码解释:
第一步:给你变量加上@Nullable @NotNull 注解,让你的在书写代码的时候就减少赋值null的错误情况
第二步:在编译代码之后会自动给你的的变量进行null值判断。
下面左边是我们的源码,右边是编译成class文件之后我们反编译得到的源码,如果想看反编译kotlin文件,可以看这里:Kotlin-20-如何查看Kotlin编译成class后的代码。
从反编译的源码中我们看到,当我们声明一个可null的变量 name1时,它不仅会给我们添加上@Nullable的注解,让我们在书写代码的时候就尽量避免赋值错误,更重要的时候,绿色框中,在编译后,它会自动帮我们添加对参数可null的判断,这样在运行的时候就不会再报错。
而当我们中声明一个变量为不可null的时候,因为我们在声明赋值阶段,就不会造成null的情况,所以运行代码的时候,一般是不会报错的,除非有一种情况会报错--->看第4节
输出结果:
0
null
4、Kotlin的null异常报错
上面我们讲了 可null的参数,编译之后会自动添加判null操作,而不可null的参数,又在声明的时候无法赋值为null,那怎么样才会出现null异常呢?
当调用Java中的null对象输出的时候会报错。
//java中的代码
public class TestNull {
String name="";
@Nullable
String name1=null;
@NotNull
String name2=null;
}
//Kotlin中的代码
fun printName(str: String) {
println(str.length)
}
fun printName1(str: String?) {
println(str?.length)
}
fun main() {
val test= TestNull()
printName1(test.name2)
printName(test.name2)
// printName(test.name1) ---报错 因为name1是可null类型,无法赋值到不可null类型上。
}
输出结果:
从输出结果看出pringName1 对null做了处理,所以不会报错,但是Kotlin中对不可null类型只是进行了抛出异常的处理,并没有做其他处理,所以这样在调用Java的null值时还是会报错。