当前位置: 首页 > news >正文

Kotlin 异常处理

文章目录

  • 什么是异常
  • 抛出异常
  • 通过异常信息解决异常
  • 捕获异常

什么是异常

我们在运行程序时,如果代码出现了语法问题或逻辑问题,会导致程序编译失败或退出,称为异常。运行结果会给出一个一长串的红色字,通常会给出异常信息(异常名、调用堆栈等)。

语法错误会直接导致编译失败,不能被代码捕获;而逻辑异常一般是在运行时抛出的,可以捕获。

语法错误:

fun main() {print("Hello Kotlin"
}
// 异常信息(部分)
e: file:///D:/Project/Kotlin/Normal/untitled/src/main/kotlin/Main.kt:2:25 Expecting ')'FAILURE: Build failed with an exception.* What went wrong:
Execution failed for task ':compileKotlin'.
> A failure occurred while executing org.jetbrains.kotlin.compilerRunner.GradleCompilerRunnerWithWorkers$GradleKotlinCompilerWorkAction> Compilation error. See log for more details

逻辑错误:

fun main() {val list = listOf(1, 2, 3)print(list[3])
}
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: Index 3 out of bounds for length 3at java.base/java.util.Arrays$ArrayList.get(Arrays.java:4266)at MainKt.main(Main.kt:4)at MainKt.main(Main.kt)

每一个异常都是一个Throwable的对象,这是一个 Java 的类。

public class Throwable implements Serializable {...
}

其实,Kotlin 中的异常来自于 Java 异常,并且使用typealias起了别名。

// Typealias.kt
package kotlin@SinceKotlin("1.1") public actual typealias Error = java.lang.Error
@SinceKotlin("1.1") public actual typealias Exception = java.lang.Exception
...

Note:Kotlin 是支持跨平台的,并且有相应的分支,Kotlin/Jvm(桌面端、安卓Android)、Kotlin/Native(原生平台,如苹果 iOS)、Kotlin/JS(网页端 JavaScript)、Kotlin/Wasm(网页端 Assembly)。
Kotlin 跨平台项目有expectactual关键字,分别用于声明全平台期望内容(expect)以及各个平台的实现内容(actual)。以上的代码前方都标记了actual,说明这是平台代码,各个平台之间代码会有所差异,Jvm 平台有 Java,而其他平台没有。

从源码中可以发现,并没有Throwable的别名,Kotlin 创建了一个自己的Throwable对象。应该是为了跨平台兼容,其他平台没有 Java 也就没有java.lang.Throwable(其他平台应该是继承自kotlin.Throwable),在实际使用中我们用的是kotlin.Throwable

// Throwable.kt
package kotlinpublic open class Throwable(open val message: String?, open val cause: Throwable?) {constructor(message: String?) : this(message, null)constructor(cause: Throwable?) : this(cause?.toString(), cause)constructor() : this(null, null)
}

抛出异常

我们可以使用throw抛出一个Throwable(不能是java.lang.Throwable),我们可以实例化Throwable或其子类,使用throw将其抛出,后续代码将不再执行。

fun main() {throw Throwable("你好,异常")print("Hello Kotlin")
}
Exception in thread "main" java.lang.Throwable: 你好,异常at MainKt.main(Main.kt:2)at MainKt.main(Main.kt)

通过异常信息解决异常

异常信息(红色字)是我们解决异常的关键。我们可以从异常信息中找到异常的名称以及抛出位置。

通常情况下,异常信息会非常长,异常信息的格式其实是固定的,异常名称、信息message值和抛出位置会在开头处,我们可以看一下下方的异常信息。

首先在开头处我们便可以得知在main这一线程(thread)中出现了一个异常(Exception in thread "main"),并且在后边跟着它的名称java.lang.Throwable,冒号:后面是异常的message值,如果messagenull,则不会有冒号及message的内容。

一般情况下,你只需要搜索第一行的信息,便会找到解决办法。当然,为了与Java 区分开,你可以在信息前加 Kotlin。

后边的就是调用堆栈,在这里可以找到异常抛出处,我们要注意那些蓝色字体(在 IDEA 中,Main.kt:2是蓝色的。Main.kt表示文件名;2表示行数,有时候会对不上),点击它光标会定位到该位置。越靠前的,就越靠近异常抛出位置。

有些时候,在中间位置也会出现类似Cause by ...的内容,也需要特别留意,可以像搜索第一行一样搜索它。

fun main() {throw Throwable("抛出异常")
}
Exception in thread "main" java.lang.Throwable: 抛出异常at MainKt.main(Main.kt:2)at MainKt.main(Main.kt)

捕获异常

有些情况下,我们并不希望异常直接抛出,例如,在安卓 Android 应用中,抛出异常将直接导致应用闪退,这是非常不好的体验。我们可以使用try-catch-finall来捕获异常。

finall可以缺省,我们先从简单的try-catch讲起。我们在try的花括号{}里可以写一段要执行的可能发生异常的代码,在后边补上catch,小括号()中需要写一个变量名称: Throwable 或其子类,表示你要捕获的异常,接着在catch花括号{}里写上捕获异常后要做什么,这里我们打印异常的消息,这是一个String?类型。

fun main() {try {// 这是一段要执行的代码println("try 1")throw Exception("异常消息")print("try 2")}catch (e: Exception) {// 如果上方代码抛出了异常// 则该地方可以尝试捕获异常print(e.message)}
}
try 1
异常消息

可以看到结果中没有异常了。如果我们去掉异常抛出语句,try中的内容会顺利执行,并且不会执行catch的内容:

fun main() {try {// 这是一段要执行的代码println("try 1")// throw Exception("异常消息")print("try 2")}catch (e: Exception) {// 如果上方代码抛出了异常// 则该地方可以尝试捕获异常print(e.message)}
}
try 1
try 2

这里需要注意,如果抛出的异常与catch圆括号中的类型不是同一类(不一定要类型相同,抛出异常也可以是圆括号中类型的子类),会捕获失败,抛出异常。这里举一个我一直以来都犯的错误,我会习惯性地把e的类型给成Exception,可是Throwable大致是分为两大类的:ExceptionError。我的声明e: Exception只对Exception及其子类生效,如果某一次它抛出了Error,将会捕获失败。

import java.awt.AWTErrorfun main() {try {// AWTError: Errorthrow AWTError("AWT 错误")}catch (e: Exception) {print(e.message)}
}
Exception in thread "main" java.awt.AWTError: AWT 错误at MainKt.main(Main.kt:6)at MainKt.main(Main.kt)

所以建议大家,还是把它的类型写为Throwable

fun main() {try {// AWTError: Errorthrow AWTError("AWT 错误")}catch (e: Throwable) {print(e.message)}
}
AWT 错误

最后是finally,它不管是否抛出异常,都会执行相应的代码:

// 将 block 函数交给 catching 函数调用
inline fun catching(block: () -> Unit) {try {// 调用 block 函数block()} catch (_: Throwable) {// 空语句需要将 e 命名为 _} finally {println("Finally")}
}fun main() {// 抛出异常catching { throw Throwable() }// 不抛出异常catching {  }
}
Finally
Finally

try-catch-finally还能用于赋值,这与ifwhen类似:

fun main() {val msg = try {throw Throwable("抛出异常")"顺利执行"}catch (e: Throwable) {"出现异常"}print(msg)
}
出现异常

相关文章:

  • 动态语言的开源编译器汇总
  • 链表的中间结点
  • Mybatis最全学习文档
  • Linux编译器-gcc或g++的使用
  • 区间预测 | Matlab实现QRCNN-BiGRU-Attention分位数回归卷积双向门控循环单元注意力机制时序区间预测
  • 数据流图(DFD)绘制规范
  • Visual 怎么编程:一场视觉与逻辑的奇妙旅程
  • 7.0 android中Service的基础知识
  • BeagleBone Black入门总结
  • 四种跨域解决方案
  • 初识C++ · 模拟实现list
  • 从零手写实现 nginx-11-文件处理逻辑与 range 范围查询合并
  • 使用Python操作Redis
  • Redis 内存回收
  • Debezium日常分享系列之:Debezium 2.6.2.Final发布
  • #Java异常处理
  • Angular2开发踩坑系列-生产环境编译
  • ECMAScript 6 学习之路 ( 四 ) String 字符串扩展
  • JavaScript创建对象的四种方式
  • JavaScript工作原理(五):深入了解WebSockets,HTTP/2和SSE,以及如何选择
  • nodejs:开发并发布一个nodejs包
  • RxJS: 简单入门
  • SAP云平台里Global Account和Sub Account的关系
  • 基于Dubbo+ZooKeeper的分布式服务的实现
  • 如何用vue打造一个移动端音乐播放器
  • 携程小程序初体验
  • 责任链模式的两种实现
  • 通过调用文摘列表API获取文摘
  • # C++之functional库用法整理
  • #我与虚拟机的故事#连载20:周志明虚拟机第 3 版:到底值不值得买?
  • (01)ORB-SLAM2源码无死角解析-(56) 闭环线程→计算Sim3:理论推导(1)求解s,t
  • (SpringBoot)第二章:Spring创建和使用
  • (备忘)Java Map 遍历
  • (附源码)ssm考试题库管理系统 毕业设计 069043
  • (附源码)ssm失物招领系统 毕业设计 182317
  • (蓝桥杯每日一题)平方末尾及补充(常用的字符串函数功能)
  • (三)Pytorch快速搭建卷积神经网络模型实现手写数字识别(代码+详细注解)
  • (万字长文)Spring的核心知识尽揽其中
  • (五十)第 7 章 图(有向图的十字链表存储)
  • (限时免费)震惊!流落人间的haproxy宝典被找到了!一切玄妙尽在此处!
  • (学习日记)2024.01.19
  • (一)spring cloud微服务分布式云架构 - Spring Cloud简介
  • (已更新)关于Visual Studio 2019安装时VS installer无法下载文件,进度条为0,显示网络有问题的解决办法
  • (译)2019年前端性能优化清单 — 下篇
  • (译)计算距离、方位和更多经纬度之间的点
  • (转载)VS2010/MFC编程入门之三十四(菜单:VS2010菜单资源详解)
  • *算法训练(leetcode)第四十五天 | 101. 孤岛的总面积、102. 沉没孤岛、103. 水流问题、104. 建造最大岛屿
  • .net core 的缓存方案
  • .net Signalr 使用笔记
  • .Net中ListT 泛型转成DataTable、DataSet
  • /etc/sudoer文件配置简析
  • @GetMapping和@RequestMapping的区别
  • @KafkaListener注解详解(一)| 常用参数详解
  • [ 第一章] JavaScript 简史
  • [100天算法】-目标和(day 79)