Kotlin之Elvis 操作符
时间:2022-04-24
本文章向大家介绍Kotlin之Elvis 操作符,主要内容包括Elvis 操作符的优先级、与 ?. 配合使用时的问题、?: 与流程控制语句的搭配使用、改良 Elvis、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。
Elvis 操作符的优先级
首先要注意到,Elvis 操作符的优先级是比较低的,特别是比加减乘除的运算符低,混合运算时一定要记得加上括号。比如:
fun <T> Iterable<T>.contentHashCode(): Int {
return fold(1) {
hash, element ->
hash * 31 + (element?.hashCode() ?: 0)
}
}
这里如果不用括号将 element?.hashCode() ?: 0 括起来,编译器就会认为这句表达式是 (hash * 31 + element?.hashCode()) ?: 0,出现编译错误。
与 ?. 配合使用时的问题
Elvis 操作符与安全调用符 ?. 配合使用时,一定要考虑到安全调用符前后是否为空,否则就会带来流程控制混乱的问题。对于任何一个下列的表达式:
val v = a?.b ?: c
因为 ?. 的优先级比 ?: 高,首先计算 a?.b,按照安全调用符的规则,如果 a == null 则结果为 null,执行 c,但如果 a.b == null,也会执行 c。也就是说,它的执行逻辑是这样的:
var temp = if(a != null) a.b else null
val v = if(temp != null) temp else c
它等价于:
val v = if(a == null || a.b == null) c else a.b
实际使用时一定要注意 ?. 前后是否都可能为 null。
?: 与流程控制语句的搭配使用
我发了一个 Kotlin 写的前序遍历二叉树的 Gist,地址在这里:PreOrderTraversing.kt,整个项目:DataStructureLearning-Kotlin 非递归遍历二叉树的代码如下:
private fun preOrderLoop(
root: Node? = this.root,
list: MutableList<Int> = mutableListOf()
): List<Int> {
val stack = ArrayDeque<Node>()
stack.push(root ?: return list)
while (stack.isNotEmpty()) {
val nodeNow = stack.pop()!!
list += nodeNow.value
nodeNow.right?.let { stack.push(it) }
nodeNow.left?.let { stack.push(it) }
}
return list
}
第二句很有意思。ArrayDeque 不能容纳 null,一旦插入 null 就会抛出 NPE,而我们的函数要求 root 为 null 时返回一个空的 List,所以这里 push() 的参数写成 root ?: return list,这句代码的逻辑如下:
if(root == null) return list
stack.push(root)
此外,Elvis 还可以配合 break 和 continue 来控制循环流程。
改良 Elvis
Elvis 操作符很方便,但只能连接表达式,我们可以写一个扩展函数来作为加强版的 Elvis 操作符。
inline infix fun <T : Any> T?.ifNull(block: (T?) -> T): T {
if (this == null) {
return block(this)
}
return this
}
使用方式:
val file = java.io.File("C:\FakeFile")
val parent = file.parent ifNull {
// do something here
}
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Yii框架模拟组件调用注入示例
- 解决laravel 表单提交-POST 异常的问题
- laravel5.0在linux下解决.htaccess无效和去除index.php的问题
- laravel返回统一格式错误码问题
- 使用 PHP Masked Package 屏蔽敏感数据的实现方法
- PHP简单实现图片格式转换(jpg转png,gif转png等)
- 在thinkphp5.0路径中实现去除index.php的方式
- Laravel5.5 手动分页和自定义分页样式的简单实现
- laravel自定义分页的实现案例offset()和limit()
- Laravel6.0.4中将添加计划任务事件的方法步骤
- Laravel 不同生产环境服务器的判断实践
- 解决thinkPHP 5 nginx 部署时,只跳转首页的问题
- Laravel 类和接口注入相关的代码
- laravel unique验证、确认密码confirmed验证以及密码修改验证的方法
- laravel 如何实现引入自己的函数或类库