Kotlin的可空性

参考链接

《Kotlin实战》

可空性是Kotlin的一个重要特性,可以帮助我们避免发生NullPointerException.

可空类型

1
2
3
int strLen(String s){
return s.length();
}

上面的Java代码如果我们传入空参数就会导致NullPointerException.

但是,和上面等价含义的Kotlin语句在编译期就能标记出错误,如下:

1
2
3
4
fun strLen(s: String) = s.length

>>> strLen(null)
ERROR: Null can not be a value of non-null type String

函数参数被声明为String类型,所以必须包含一个非null的String实例,这样在Kotlin中就保证了不会在运行时抛出NullPointerException.

和Java不同的是Kotlin中声明的变量默认都是非null的,如果可空则需要追加?号来声明。

Type? = Type or null

1
fun strLen(s: String?) = s.length

上面这行代码编译器会直接标记错误,因为s.length的s可能会是null

另外不能将可空类型的值赋值给非空类型的变量。

1
2
3
4
var x: String? = null
var y: String = x

ERROR: Type mismatch: inferred type is String? but String was expected

安全调用运算符:”?.”

等价于: if not null

1
2
3
4
5
if (s != null) s.toUpperCase() else null
|
//等价于
|
s?.toUpperCase()

通过该运算符可以很方便的进行非null判断了,这个在实际开发中很实用。

Elvis运算符:”?:”

等价于:if not null and else

1
2
3
fun foo(s: String){
val t: String = s ?: ""
}

如果s为null,结果会是一个””字符串,使用该运算符可以将null替换为一个默认值。

安全转换:”as?”

as?运算符尝试把值转换成指定的类型,如果值不是合适的类型就返回null

1
2
3
4
5
6
7
8
class Person(val firstName: String, val lastName: String){

override fun equals(o: Any?): Boolean{
val otherPerson = o as? Person ?: return false
return otherPerson.fristName == firstName &&
otherPerson.lastName = lastName
}
}

非空断言:”!!”

1
2
3
4
5
6
7
8
fun ignoreNulls(s: String?){
val sNotNull: String = s!!
println(sNotNull.length)
}

>>> ignoreNulls(null)

Exception in thread "main" kotlin.KotlinNullPointerException

!!运算符可以把任何类型转换成非空类型,如果对null值做非空断言则会抛出异常。

“let”函数

1
2
3
4
5
fun sendEmailTo(email: String){
/* send Email */
}

if(email != null) sendEmailTo(email)

上面我们显式的判断了email不为null,可以使用let函数达到同样的目的。

1
email?.let {email -> sendEmailTo(email)}

let函数可以让我们隐式的判断非空参数并传递给调用函数。

延迟初始化属性

Kotlin中要求在构造方法中初始化所有属性,如果某个属性是非空类型,你就必须提供非空的初始化值,否则,你就必须使用可空类型。

但是有时候我们在构造中无法进行初始化,但是又需要在另一个入口处初始化并且让其是非空类型(非空类型可以不用重复判断是否为空)。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class MyService{
fun performAction(): String = "foo"
}

class MyTest{

private var myService: MyService? = null

@Before fun setUp(){
myService = MyService()
}

@Test fun testAction(){
Assert.aseertEquals("foo", myService!!.performAction())
}
}

为了解决上面的问题,可以将myService属性声明为延迟初始化属性,使用lateinit修饰符。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class MyService{
fun performAction(): String = "foo"
}

class MyTest{

private lateinit var myService: MyService

@Before fun setUp(){
myService = MyService()
}

@Test fun testAction(){
Assert.assertEquals("foo", myService.performAction())
}
}

可空类型的扩展

1
2
3
4
var str: String? = null
println(str.isNullOrBlank())

>>> true

你是不是感到奇怪,为什么可空类型直接可以调用方法呢?

事实上isNullOrBlack()是String类的一个扩展函数,扩展函数可以允许接收者为null调用,并在函数中处理null,只有扩展函数能这样做。

和isNullOrBlack()类似的isEmptyOrNull也是String的扩展函数。

1
fun String?.isNullOrBlack(): Boolean = this == null || this.isBlank()

当你为一个可空类型(以?结尾)定义扩展函数时,意味着你可以用可空的值调用这个函数,函数体中的this可能为null。

类型参数的可空性

1
2
3
fun <T: Any> printHashCode(t: T){
println(t?.hashCode())
}

在Kotlin中所有泛型类和泛型函数的类型参数默认是可空的。上面的类型T默认是可空的。

可空性和Java

Java中的@Nullable String 等价于 Kotlin中的 String?
Java中的@NotNull String 等价于 Kotlin中的 String

评论

Ajax Android AndroidStudio Animation Anroid Studio AppBarLayout Banner Buffer Bulma ByteBuffer C++ C11 C89 C99 CDN CMYK COM1 COM2 CSS Camera Raw, 直方图 Chrome ContentProvider CoordinatorLayout C语言 DML DOM Dagger Dagger2 Darktable Demo Document DownloadManage Element Error Exception Extensions File FileProvider Fresco GCC Git GitHub GitLab Gradle Groovy HTML5 Handler HandlerThread Hexo Hybrid I/O IDEA IO ImageMagick IntelliJ Intellij Interpolator JCenter JNI JS Java JavaScript JsBridge Kotlin Lab Lambda Lifecycle Lint Linux Looper MQTT MVC MVP Maven MessageQueue Modbus Momentum MySQL NDK NIO NexT Next Nodejs ObjectAnimator Oracle VM Permission PhotoShop Physics Python RGB RS-232 RTU Remote-SSH Retrofit Runnable RxAndroid RxJava SE0 SSH Spring SpringBoot Statubar Task Theme Thread Tkinter UI UIKit UML VM virtualBox VS Code ValueAnimator ViewPropertyAnimator Web Web前端 Workbench api apk bookmark by关键字 compileOnly css c语言 databases demo hexo hotfix html iOS icarus implementation init jQuery javascript launchModel logo merge mvp offset photos pug query rxjava2 scss servlet shell svg tkinter tomcat transition unicode utf-8 vector virtual box vscode 七牛 下载 中介者模式 串口 临潼石榴 主题 书签 事件 享元模式 仓库 代理模式 位运算 依赖注入 修改,tables 光和色 内存 内核 内部分享 函数 函数式编程 分支 分析 创建 删除 动画 单例模式 压缩图片 发布 可空性 合并 同向性 后期 启动模式 命令 命令模式 响应式 响应式编程 图层 图床 图片压缩 图片处理 图片轮播 地球 域名 基础 增加 备忘录模式 外观模式 多线程 大爆炸 天气APP 太白山 头文件 奇点 字符串 字符集 存储引擎 宇宙 宏定义 实践 属性 属性动画 岐山擀面皮 岐山肉臊子 岐山香醋 工具 工厂模式 年终总结 开发技巧 异常 弱引用 恒星 打包 技巧 指针 插件 摄影 操作系统 攻略 故事 数据库 数据类型 数组 文件 新功能 旅行 旋转木马 时序图 时空 时间简史 曲线 杂谈 权限 枚举 架构 查询 标准库 标签选择器 样式 核心 框架 案例 桥接模式 检测工具 模块化 模板引擎 模板方法模式 油泼辣子 泛型 洛川苹果 浅色状态栏 源码 瀑布流 热修复 版本 版本控制 状态栏 状态模式 生活 留言板 相册 相对论 眉县猕猴桃 知识点 码云 磁盘 科学 笔记 策略模式 类图 系统,发行版, GNU 索引 组件 组合模式 结构 结构体 编码 网易云信 网格布局 网站广播 网站通知 网络 美化 联合 膨胀的宇宙 自定义 自定义View 自定义插件 蒙版 虚拟 虚拟机 补码 补齐 表单 表达式 装饰模式 西安 观察者模式 规范 视图 视频 解耦器模式 设计 设计原则 设计模式 访问者模式 语法 责任链模式 贪吃蛇 转换 软件工程 软引用 运算符 迭代子模式 适配器模式 选择器 通信 通道 配置 链表 锐化 错误 键盘 闭包 降噪 陕西地方特产 面向对象 项目优化 项目构建 黑洞
Your browser is out-of-date!

Update your browser to view this website correctly. Update my browser now

×