kotlin:null和相等判断和this代表了什么
2024-03-12 本文已影响0人
爱你为你做饭
本文章是kotlin中几个小的知识点的总结整理:包括判空,==和===(相等判断),还有this的用法三个知识点。
空指针异常及其kotlin的判空操作:
- 空指针异常:无论是在java还是kotlin,访问一个对象的属性或者函数都得是这个对象已存在的情况下,若这个对象不存在就会出现空指针异常,项目崩溃是项目开发中非常常见的一个异常,对异常处理或者抓取或者抛出,但是null通常是逻辑出现问题出现的一个崩溃异常,具有很大的隐蔽性。java中通常是根据经验进行判空再执行,比如接口字段,集合的item使用等,kotlin针对null异常做了很多补充措施尽可能的去避免null异常。
- kotlin对null做的处理
- kotlin中声明的属性通常都是非null的,在属性的使用的时候kotlin编译的时候会自动加上null校验。
-
kotlin若要某个属性支持null则声明属性的时候后面用关键字?后缀,即A?表示声明的属性A支持null。
非null属性
可null属性
- kotlin中出现空指针异常的几种场景
- 明确调用 throw NullPointerException()
- 使用 !! 操作符:会将可null属性强制转为非null属性,若属性为null则会直接抛出空指针异常。
- 初始化过程中存在数据不一致, 比如:
- 在构造器中可以访问到未初始化的 this, 并且将它传递给了其他代码, 然后在其他代码中使用了这个未初始化的 this (也就是所谓的 "leaking this" 警告);
- 基类的构造器调用了 open 的成员函数, 但这个成员函数在子类中的实现使用了未初始化的状态数据.
- Java 互操作:
- 试图对一个 平台类型 的
null
引用访问其成员函数. - 用于 Java 互操作的泛型类型的可空性存在问题. 比如, 一段 Java 代码可能向一个 Kotlin MutableList<String> 中添加一个 null 值, 因此, 对这种情况应该使用 MutableList<String?>.
- 外部 Java 代码导致的其他问题.
- 试图对一个 平台类型 的
- kotlin不知道属性是否为null的情形下的几种判空操作
-
和java类似,通过if语句对属性进行判空校验,即:
if语句 - 通过?.(安全调用)调用属性,属性为null不执行后面的语句返回null,属性不为null才会执行后面的语句。
- 安全调用在属性的链式调用上非常有用,特别是接口中返回的对象深度嵌套使用安全调用则非常完美,即:bob?.department?.head?.name:每一个层级上对象为null则返回null后续不再执行。
- ?.和let函数的完美结合,对于函数体中多条语句需要使用属性的场景是极好的解决方法,let可以将自己对象作为lambda表达式的参数传递,?.可以判空校验,很简单的就可以实现属性非null的用于下面的多语句。
-
在赋值运算的左侧也可以使用安全调用. 这时, 如果链式安全调用中的任何一个接受者为 null, 赋值运算就会被跳过, 完全不会对赋值运算右侧的表达式进行计算:
属性的安全调用
- 通过对属性对象的扩展函数来处理属性为null的场景下的逻辑处理避免其出现null指针异常,比如扩展对象的toString函数,当其属性为null的时候直接输出null字符串,简单说就是通过扩展函数扩充了属性为null的场景。此种方法适合与对象的函数的多处调用,一处修改多处使用即可。
-
Elvis操作符:操作符使用?:代替,简单说就是使用这个操作符简化了上面提到的if语句,val l = b?.length ?: -1 (b为null返回-1,不为null返回length,需要注意的是只有属性为null的时候才会计算后面的表达式)
?:操作符 - !!操作符:上面有介绍,即将可null属性强行修改为非null属性,若属性为null则抛出空指针异常。
-
安全的类型转换:kotlin提供的固定语法:用于类的转换,在转换过程中若类型不一致则返回null进而避免了类的强转异常:
安全的类型转换 -
通过集合高阶函数(filterNotNull)可以快速过滤出集合的非null item集合,即:
filterNotNull函数
-
相等判断(==和===):
- 结构相等:属性类型及其内容相等被称为结构相等,用操作符==和!=描述,且kotlin上的==和java中的equals相等,即kotlin可以使用==判断两个对象是否结构相等。
- 引用相等:即校验两个属性的内存地址来判断两个属性是否相等,使用操作符===和!==描述 针对基础变量===等同于==。
- 结构相等和引用相等:对比的参照不一样也就造成两者的对比没有可比性,前者对比的是属性类型和内容,后者是属性的内存地址,比如同一个类型的同一个属性不同对象,前者结构相等但是引用不相等(类型和内容一致,内存地址不一致)
- 数组的相等的比较:要比较两个数组是否包含相同顺序的相同元素, 请使用 contentEquals() 。
this的代表意义:
- 在 类 的成员函数中,
this
指向这个类的当前对象实例. - 在 扩展函数 中, 或 带接收者的函数字面值(function literal) 中,
this
代表调用函数时, 在点号左侧传递的 接收者 参数. - 如果 this 没有限定符, 那么它指向 包含当前代码的最内层范围. 如果想要指向其他范围内的 this, 需要使用 标签限定符:
- 带有限定符的this:为了访问更外层范围(比如 类, 或 扩展函数, 或有标签的 带接受者的函数字面值)内的
this
, 你可以使用this@label
, 其中的@label
是一个 标签, 代表你想要访问的this
所属的范围:
带有标签的this操作符