Swift 异常处理

异常处理:
一、引入
二、异常表示
三、异常处理-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)
}