1、Kotlin
有两种类型:一个是非空引用类型,一个是可空引用类型。
2、对于可空引用,如果希望调用它的成员变量或者成员函数,直接调用会出现编译错误,有三种方法可以调用:
(1)在调用前,需要先检查,因为可能为null
(2)使用b?.length
的形式调用,如果b为null
,返回null
,否则返回b.length
(3)使用b!!.length()
的形式调用,如果b为null
,抛出空指针异常NullPointerException
,否则返回b.length
可空类型和非可空类型
在kotlin
中变量默认是不能为空的
var string :String = null
这个是会报错的。
变成
var string :String? = null
在一个参数后面加入问号?
说明这是一个可空变量。
null
和空
之间的区别
-
null
是指对象的指针
指向了一个空的地方
,这个指针指向了null
,是安全的。 - 对象为空其实是指的一个对象没能初始化,在未声明一个变量
lateinit
的时候,JVM
会自动为变量分配一个内存地址,但是没有初始化,如果在这个时候访问这个对象会导致访问空指针出现NPE
问题NullPointerException
Kotlin
的类型系统
旨在从我们的代码中消除NullPointerException
。NPE
的唯一可能的原因可能是
- 显式调用
throw NullPointerException()
- 使用了下文描述的
!!
操作符 - 外部
Java
代码导致的 - 对于初始化,有一些数据不一致(如一个未初始化的
this
用于构造函数的某个地方)
类型系统可以区分一个引用是否可以容纳 null (可空引用)还是不能容纳(非空引用)。 例如,String 类型的常规变量不能容纳 null。
//如果a没有指明可以为空,那么赋null值的时候编译就不通过,所以你调用` a` 的方法或者访问属性,它保证不会导致 NPE,这样你就可以放心地使用:
var a: String = "abc"
a = null // 编译错误
var a: String = "abc";
Log.i(TAG, "a.length ${a.length}")
如果要允许为空,我们可以使用
?
声明一个变量为可空字符串,写作?
//如果变量b声明了可以为空值,那么赋null值得时候编译通过
var b: String? = "abc"
b = null // ok
//但是如果你想访问 b 的同一个属性,那么这是不安全的,并且编译器会报告一个错误:
//当b=null的时候,编译器不允许取长度值;
//var b=b.length //编译报错!!!
var b: String?;
b = null;
//Log.i(TAG,"b.length ${b.length}")//编译错误
?.
安全调用操作符:
如果
b
非空,就返回b.length
,否则返回null
,这个表达式的类型是Int?
。
var result2 = b?.length;
student?.class?.school
Elvis
》?:
操作符
如果
?:
左侧表达式非空,elvis
操作符就返回其左侧表达式,否则返回右侧表达式。 请注意,当且仅当左侧为空时,才会对右侧表达式求值。等同if else
。因为throw
和return
在Kotlin
中都是表达式,所以它们也可以用在elvis
操作符右侧
var result = b?.length ?:"b为null";
//等价于
var result = if (b != null) b.length else "b为null"
!!
操作符
b!!
,这会返回一个非空的b
值 (例如:在我们例子中的String
)或者如果 b 为空,就会抛出一个NPE
异常:
b!!非空断言运算符(!!)将任何值转换为非空类型,若该值为空则抛出异常。
val l = b!!.length
as?
类型转换异常ClassCastException
安全类型转换,如果类型转换不成功,就会返回null,而不是抛出ClassCastException异常。
var a: Long = 1
val b: Int? = a as Int // java.lang.ClassCastException
//更改为:
var a: Long = 1
var b: Int? = a as? Int