Kotlin实用语法糖空安全类型转换及相等性判断

前言

Kotlin被Google官方认为是Android开发的一级编程语言。今天,我将主要讲解,关于Kotlin的一些实用语法糖,主要包括:

  • 范围使用:in、downTo、step、until
  • 类型检查 & 转换:is、智能转换、as
  • 相等性:equals()、== 、 ===
  • 空安全

1. 范围使用

主要用于表示范围,主要包括:in、downTo、step、until

/**
 *  1. in
 *  作用:在...范围内
 **/
// 表示:若i在1-5范围内,则执行下面代码
// 注:闭区间,[1,5]
if (i in 1..5) {
    println("i 在 1-5 内")
}
// 表示:若i不在1-5范围内,则执行下面代码
// !in表示不在...范围内
if (i !in 1..5) {
    println("i 不在 1-5 内")
}
/**
 *  2. until
 *  作用:表示开区间
 **/
// 输出1234
for (i in 1 until 5) {
 println(i)
}
/**
 *  3. downTo
 *  作用:倒序判断
 **/
 for (i in 5 downTo 1) {
 	println(i)
 }
/**
 *  4. step
 *  作用:调整步长
 **/
// 设置步长为2,顺序输出1、3、5
for (i in 1..5 step 2) println(i)
// 设置步长为2,倒序输出5、3、1
for (i in 1 downTo 5 step 2) println(i) 

2. 类型检查 & 转换

包括:is、智能转换 和 as

/**
 *  1. is
 *  作用:判断一个对象与指定的类型是否一致
 **/
// 判断变量a的数据类型是否是String
var a: Any = "a"
if (a is String) {
    println("a是String类型")
}
if (a !is Int) {
    println("a不是Int类型")
}
/**
 *  2. 智能转换
 *  说明: kotlin不必使用显式类型转换操作,因为编译器会跟踪不可变值的is检查以及显式转换,并在需要时自动插入(安全的)转换
 **/
 var a: Any = "a"
if (a is String) {
    println("a是String类型")
    println(a.length) // a 自动转换为String类型
    //输出结果为:1
}
// 反向检查: a自动转换为String类型
if (a !is String) {
	print(a.length)
}
// 在 && 和 || 的右侧也可以智能转换:
// `&&` 右侧的 a 自动转换为String
if (a is String && a.length > 0)
// `||` 右侧的 a 自动转换为String
if (a is String || a.length > 0)
// 在when表达式和while循环里也能智能转换:
when(a){
    is String -> a.length
    is Int -> a + 1
}
// 需要注意:当编译器不能保证变量在检查和使用之间不可改变时,智能转换不能用。智能转换能否适用根据以下规则:
// 1. val 局部变量——总是可以,局部委托属性除外;
// 2. val 属性——如果属性是 private 或 internal,或者该检查在声明属性的同一模块中执行。智能转换不适用于 open 的属性或者具有自定义 getter 的属性;
// 3. var 局部变量——如果变量在检查和使用之间没有修改、没有在会修改它的 lambda 中捕获、并且不是局部委托属性;
// 4. var 属性——决不可能(因为该变量可以随时被其他代码修改)
/**
 *  3. 强制类型转换:as
 **/
var any: Any = "abc"
var str: String = any as String
// 强制类型转换是不安全的,若类型不兼容则会抛出一个异常
var int: Int = 123
var str: String = int as String
// 抛出ClassCastException
/**
 *  4. 可空转换操作符:as?
 *  作用:null不能转换为String,因该类型不是可空的,此时使用可空转换操作符as?
 **/
var str = null
var str2 = str as String
// 抛出TypeCastException
// 使用安全转换操作符as?可以在转换失败时返回null,避免了抛出异常。
var str = null
var str2 = str as? String
println(str2) //输出结果为:null

3. 相等性判断

在Kotlin中,存在结构相等 & 引用相等 两种相等判断。

/**
 *  1. 结构相等:equals()或 ==
 *  作用:判断两个结构是否相等
 **/
var a = "1"
var b = "1"
if (a.equals(b)) {
    println("a 和 b 结构相等")
    // 输出结果为:a 和 b 结构相等
}
var a = 1
var b = 1
if (a == b) {
    println("a 和 b 结构相等")
    // 输出结果为:a 和 b 结构相等
}
/**
 *  2. 引用相等:===
 *  作用:判断两个引用是否指向同一对象
 */
// 设置一个类如下
data class User(var name: String, var age: Int)
// 设置值
var a = User("Czh", 22)
var b = User("Czh", 22)
var c = b
var d = a
// 对比两个对象的结构
if (c == d) {
    println("a 和 b 结构相等")
} else {
    println("a 和 b 结构不相等")
}
// 对比两个对象的的引用
if (c === d) {
    println("a 和 b 引用相等")
} else {
    println("a 和 b 引用不相等")
}
// 输出结果:
a 和 b 结构相等
a 和 b 引用不相等

4. 空安全

  • 在Java中,NullPointerException异常十分常见
  • 而Kotlin的优点则是可以尽可能避免执行代码时出现的空指针异常
/**
 *  1. 可空类型与非空类型
 *  在Kotlin中,有两种情况最可能导致出现NullPointerException
 **/
// 情况1:显式调用 throw NullPointerException()
// 情况2:使用!! 操作符
// 说明:!!操作符将任何值转换为非空类型,若该值为空则抛出异常
var a = null
a!!
// 抛出KotlinNullPointerException
// 情况3:数据类型不能为null
// 在 Kotlin 中,类型系统区分一个引用可以容纳 null (可空引用) 和 不能容纳(非空引用)
// 如:String类型变量不能容纳null
// 若要允许为空,可声明一个变量为可空字符串:在字符串类型后面加一个问号?
对于String,则是写作:String?
var b: String? = "b"
b = null
/**
 *  2. 安全调用操作符
 *  作用:表示如果若不为null才继续调用
 **/
 b?.length
 // 表示:若b不为null,才调用b.length
 // 注:安全调用符还可以链式调用
 a?.b?.c?.d
 // 假设a不为null,才继续往下调用,以此类推
 // 若该链式调用中任何一个属性为null,整个表达式都会返回null。
 // 若只对非空值执行某个操作,可与let一起使用
a?.b?.let { println(it) }

至此,关于Kotlin的入门语法讲解完毕,更多关于Kotlin空安全类型转换相等性判断的资料请关注我们其它相关文章!

时间: 2022-06-20

Kotlin基本数据类型详解

目录 1. kotlin 数值型 2. kotlin 布尔型 3. kotlin 字符型 4. kotlin 字符串 5. kotlin 可空类型 6. kotlin 类型相互转换 7. kotlin 二进制.十六进制 1. kotlin 数值型 fun main() { // 整数型 val a: Byte = 1 // 8位 val b: Short = 2 // 16位 val c: Int = 3 // 32 val c2 = 3 // 默认整型 Int val d: Long = 4

java协程框架quasar和kotlin中的协程对比分析

目录 前言 快速体验 添加依赖 添加javaagent 线程VS协程 协程代码 多线程代码 协程完胜 后记 前言 早就听说Go语言开发的服务不用任何架构优化,就可以轻松实现百万级别的qps.这得益于Go语言级别的协程的处理效率.协程不同于线程,线程是操作系统级别的资源,创建线程,调度线程,销毁线程都是重量级别的操作.而且线程的资源有限,在java中大量的不加限制的创建线程非常容易将系统搞垮.接下来要分享的这个开源项目,正是解决了在java中只能使用多线程模型开发高并发应用的窘境,使得java也能

Kotlin可见性修饰符详解

目录 一. 四种修饰符的说明 二.在不同场景下各个修饰符声明的范围讲解 2.1.在包(package)中声明,即顶层声明的情况 2.1.1.测试同一个文件中声明不同的修饰符的范围 2.1.2.测试不同文件中声明不同的修饰符的范围 2.1.3.小结 2.2.在类(class)和接口(interface)中声明的情况 2.2.1.测试在类中声明的情况 2.2.2.测试在接口中声明的情况 2.2.3.小结 2.3.在构造函数中声明的情况 2.4.局部声明的情况 三.与Java中的可见性修饰符的对比 一

Kotlin空安全空类型浅谈

目录 正文 一.判断可空变量是否为null的方式 二.涉及到返回可空值的函数使用方法,在返回类型后同时加? 三.let操作符 四.Evils操作符(?: !! as) 总结: 正文 首先非空变量和可空变量的区别: // name为不可为空的变量, 不能赋值为null ,若有判断 if(name==null) 无意义,因为肯定不为null var name : String = "" // role 为可空变量 var role : String? = "" 非空变量

Kotlin操作符重载实例详解

目录 算数运算操作符重载 复合运算操作符重载 一元运算操作符重载 比较操作符重载 集合和区域的约定 迭代运算符重载 解构声明 总结 算数运算操作符重载 在kotlin中我定义一个类 data class Point(val x: Int, val y: Int) 然后实例化两个对象 val p1 = Point(3,5) val p2 = Point(5,7) 想表示p1的元素x加上p2的元素x,p1的元素y,加上p2的元素y.然后输出一个p3. val p3 = Point(p1.x + p2

Kotlin 单例实例详解

Kotlin 单例实例详解 单例的实现方法,可以通过同伴对象,或者 lazy. 示例: class Hello private constructor() { companion object { val instance = Hello() } } 通过 lazy 实现 class Hello private constructor() { private object Holder { val INSTANCE = Hello() } companion object { val insta

Kotlin 基础语法实例详解

Kotlin 基础语法实例详解 包 定义和引入Java一样,在文件开头, 行结束不需要" ; " package com.test.hello import android.os.Bundle 变量 只读变量,val 开头,初始化后不能再赋值,相当于Java的 final 变量 val a: Int = 1 val b = 1 //类型自动推断为Int val c: Int //没有初始化时必须指定类型 c = 1 //初始化 可变变量, var 关键字开头 var x = 10 x

C++中函数重载实例详解

C++中函数重载实例详解 函数重载: 1.具有相同的名称,执行基本相同的操作,但是使用不同的参数列表. 2.函数具有多态性. 3.编译器通过调用时参数的个数和类型确定调用重载函数的哪个定义. 4.只有对不同的数据集完成基本相同任务的函数才应重载. 函数重载的优 点 1.不必使用不同的函数名 2.有助于理解和调试代码 3.易于维护代码 接下来直接上代码: #include <iostream> using namespace std ; void say_hello(void) { cout &

Kotlin 基本语法实例详解

基本语法示例 实例代码: package com.stone.basic.syntax /** * desc : * author: stone * email : aa86799@163.com * time : 27/05/2017 11 01 */ class BasicSyntax { //Function having two Int parameters with Int return type: public fun sum(a: Int, b: Int): Int {//访问修饰

Kotlin 语言中调用 JavaScript 方法实例详解

Kotlin 语言中调用 JavaScript 方法实例详解 Kotlin 已被设计为能够与 Java 平台轻松互操作.它将 Java 类视为 Kotlin 类,并且 Java 也将 Kotlin 类视为 Java 类.但是,JavaScript 是一种动态类型语言,这意味着它不会在编译期检查类型.你可以通过动态类型在 Kotlin 中自由地与 JavaScript 交流,但是如果你想要 Kotlin 类型系统的全部威力 ,你可以为 JavaScript 库创建 Kotlin 头文件. 内联 J

JavaScript 中调用 Kotlin 方法实例详解

JavaScript 中调用 Kotlin 方法实例详解 Kotlin 编译器生成正常的 JavaScript 类,可以在 JavaScript 代码中自由地使用的函数和属性 .不过,你应该记住一些微妙的事情. 用独立的 JavaScript 隔离声明 为了防止损坏全局对象,Kotlin 创建一个包含当前模块中所有 Kotlin 声明的对象 .所以如果你把模块命名为 myModule,那么所有的声明都可以通过 myModule 对象在 JavaScript 中可用.例如: fun foo() =

Kotlin开发Android应用实例详解

Kotlin开发Android应用实例详解 相关文章:关于Kotlin语言的基础介绍: http://www.jb51.net/article/114086.htm 我们简单的知道了Kotlin这门新语言的优势,也接触了一些常见的语法及其简单的使用,相信你会对它有浓厚的兴趣,暂且理解为对它感兴趣吧,哈哈哈.那么,我们该如何在Android中应用这门新的语言呢?今天的这篇文章带你学习使用Kotlin开发Android应用,并对比我们传统语言Java,让你真真切切的感受到他的美和优雅. 配置 项目g

spring boot + jpa + kotlin入门实例详解

spring boot +jpa的文章网络上已经有不少,这里主要补充一下用kotlin来做. kotlin里面的data class来创建entity可以帮助我们减少不少的代码,比如现在这个User的Entity,这是Java版本的: @Entity public class User { @Id @GeneratedValue(strategy = GenerationType.AUTO) private long id; private String firstName; private S

Kotlin 内联函数详解及实例

Kotlin 内联函数详解及实例 概述 在说内联函数之前,先说说函数的调用过程. 调用某个函数实际上将程序执行顺序转移到该函数所存放在内存中某个地址,将函数的程序内容执行完后,再返回到转去执行该函数前的地方.这种转移操作要求在转去前要保护现场并记忆执行的地址,转回后先要恢复现场,并按原来保存地址继续执行.也就是通常说的压栈和出栈.因此,函数调用要有一定的时间和空间方面的开销.那么对于那些函数体代码不是很大,又频繁调用的函数来说,这个时间和空间的消耗会很大. 那怎么解决这个性能消耗问题呢,这个时候

Kotlin 的注解类详解及实例

Kotlin 的注解类详解及实例 注解声明 注解是将元数据附加到代码的方法.要声明注解,请将 annotation 修饰符放在类的前面: annotation class Fancy 注解的附加属性可以通过用元注解标注注解类来指定: @Target 指定可以用 该注解标注的元素的可能的类型(类.函数.属性.表达式等): @Retention 指定该注解是否 存储在编译后的 class 文件中,以及它在运行时能否通过反射可见 (默认都是 true): @Repeatable 允许 在单个元素上多次