异常处理:
一、引入
二、异常表示
三、异常处理-4种方式
四、指定清理操作
涉及关键字:Error throw throws try do catch
引入
比如你设计的程序需要读取电脑中的某个文件,以下代码当访问文件出现问题时,当前没法清楚的描述异常,无法定位错误的原因。所以如何来描述异常呢?
```
func readFileContent(filePath : String) -> String? {
// 1.filePath为""
if filePath == "" {
return nil
}
// 2.filepath有值,但是没有对应的文件
if filePath != "/User/Desktop/123.plist" {
return nil
}
// 3.取出其中的内容
return "123"
}
readFileContent(filePath: "abc")
```
描述异常
参考官网
在Swift里,错误用遵循 Error 协议的类型的值来表示;
Error是一个空的protocol,它唯一的功能,就是告诉Swift编译器,某个类型用来表示一个错误。
通常,我们使用一个enum来定义各种错误的可能性
抛出一个错误用throw
通过合理定义异常,改进上述代码
```
// 1.定义异常
enum FileReadError : Error {
case FileISNull
case FileNotFound
}
// 2.改进方法,让方法抛出异常
func readFile(filePath : String) throws -> String {
// 1.filePath为""
if filePath == "" {
throw FileReadError.FileISNull
}
// 2.filepath有值,但是没有对应的文件
if filePath != "/User/Desktop/123.plist" {
throw FileReadError.FileNotFound
}
// 3.取出其中的内容
return "123"
}
```
异常处理
抛出异常后,也就是异常出现后,怎么处理呢?
用throwing 函数传递错误
try?方式
最终返回结果为一个可选类型。如果出现了异常,则返回一个nil.没有异常,则返回对应的值——》不处理异常
try!方式
告诉系统该方法没有异常,一旦如果出现了异常,则程序会直接崩溃
do catch(建议)
```
var result = try? readFile(filePath: "abc")
var result = try! readFile(filePath: "abc")
do{
try readFile(filePath: "abc")
}catch{//内置变量error
print(error)
}
```
指定清理
在java中,我们处理异常使用 try catch finally 。不管有没有出错,我们一般把必须要执行的代码放在finally里。比较典型的一个场景是数据库的操作,不管是否操作成功,最后要close 释放资源。
在swift中,如果想要defer语句来实现。
defer语句调用时机:将离开当前代码块时执行,可以用它在异常中进行扫尾工作,比如关闭IO流,释放资源等
案例一:掌握defer执行时机
```
do{
defer{
print("释放资源11")
print("释放资源12")
}
print("test1")
try readFile(filePath: "/User/Desktop/123.plist") //如果路径为空,执行顺序如何?
print("test2")
}catch{//内置变量error
print(error)
}
```
案例二:多个defer语句执行顺序
```
do{
defer{
print("释放资源11")
print("释放资源12")
print("释放资源13")
}
defer{
print("释放资源21")
}
defer{
print("释放资源33")
}
try readFile(filePath: "/User/Desktop/123.plist")
}catch{//内置变量error
print(error)
}
```