• Swift中的Error处理


     

    错误类型

    1、开发过程常见的错误

            语法错误(编译报错)

            逻辑错误

            运行时错误(可能会导致闪退,一般也叫做异常),也是我们今天讲的重点

    自定义错误

    1、Swift中可以通过Error协议自定义运行时的错误信息

    1. enum SomeError: Error {
    2. case illegalArg(String)
    3. case outOfBounds(Int, Int)
    4. case outOfMemory
    5. }

    2、函数内部通过throw抛出自定义Error,可能会抛出Error的函数必须加上throws声明

    1. func divide(_ num1: Int, _ num2: Int) throws -> Int {
    2. if num2 == 0 {
    3. throw SomeError.illegalArg("0不能作为除数")
    4. }
    5. return num1 / num2
    6. }

    3、需要使用try调用可能会抛出Error的函数

    4、可以使用do-catch捕捉Error

    1. func test() {
    2. do {
    3. try divide(20, 0)
    4. } catch let SomeError.illegalArg(msg) {
    5. print("参数异常:", msg)
    6. } catch let SomeError.outOfBounds(size, index) {
    7. print("下标越界:", "size = \(size), index = \(index)")
    8. } catch SomeError.outOfMemory {
    9. print("内存溢出")
    10. } catch {
    11. print("其他错误")
    12. }
    13. }

    5、抛出Error后,try下一句直到作用域结束的代码都停止运行

    处理Error

    1、处理Error的2种方式

            a、通过do-catch捕捉Error

            b、不捕捉Error,在当前函数增加throws声明,Error将自动抛给上层函数

    1. func test() throws {
    2. print(try divide(200, 0))
    3. }

    如果最顶层函数(main函数)依然没有捕捉Error,那么程序将终止

    以下是几种error的处理方法:

    1. func test() throws {
    2. do {
    3. print(try divide(200, 0))
    4. } catch let error as SomeError {
    5. print(error)
    6. }
    7. }
    1. func test() throws {
    2. do {
    3. print(try divide(200, 0))
    4. } catch is SomeError {
    5. print("SomeError")
    6. }
    7. }
    1. do {
    2. try divide(20, 0)
    3. } catch let error {
    4. switch error {
    5. case let SomeError.illegalArg(msg):
    6. print("参数异常:", msg)
    7. default:
    8. print("其他异常")
    9. }
    10. }

    我们通过一个例子来总结一下处理Error的2种方式:

    1. override func viewWillAppear(_ animated: Bool) {
    2. super.viewWillAppear(animated)
    3. try test0()
    4. }
    5. func test0() throws -> Void {
    6. try test1()
    7. }
    8. func test1() throws -> Void {
    9. try test2()
    10. }
    11. func test2() throws -> Void {
    12. do {
    13. print(try divide(200, 0))
    14. } catch is SomeError {
    15. print("This is SomeError")
    16. }
    17. }

    try?、try!

    1、可以使用try?、try!调用可能会抛出Error的函数,这样就不用去处理Error

    1. func test() -> Void {
    2. print("1")
    3. var result1 = try? divide(20, 10) // Optional(2), Int?
    4. var result2 = try? divide(20, 0) // nil
    5. var result3 = try! divide(20, 10) // 2, Int
    6. print("2")
    7. }

    2、a、b是等价的

    1. var a = try? divide(20, 0)
    2. var b: Int?
    3. do {
    4. b = try divide(20, 0)
    5. } catch {
    6. b = nil
    7. }

    rethrows

    1、rethrows表明:函数本身不会抛出错误,但调用闭包参数抛出错误,那么它会将错误向上抛

    1. func exec(_ fn: (Int, Int) throws -> Int, _ num1: Int, _ num2: Int) rethrows -> Void {
    2. print(try fn(num1, num2))
    3. }

    defer

    1、defer语句:用来定义以任何方式(抛错误、return等)离开代码块前必须要执行的代码

            defer语句将延迟到当前作用域结束之前执行

    1. func open(_ filename: String) -> Int {
    2. print("open")
    3. return 1
    4. }
    5. func close(_ file: Int) -> Void {
    6. print("close")
    7. }
    8. func processFile(_ filename: String) throws -> Void {
    9. let file = open(filename)
    10. defer {
    11. close(file)
    12. }
    13. try divide(20, 0)
    14. }

    2、defer语句的执行顺序与定义顺序相反

    1. func fn1() -> Void {
    2. print("fn1")
    3. }
    4. func fn2() -> Void {
    5. print("fn2")
    6. }
    7. func testDefer() -> Void {
    8. defer {
    9. fn1()
    10. }
    11. defer {
    12. fn2()
    13. }
    14. } //fn2 fn1
  • 相关阅读:
    本地引入 Axios 报错
    jmeter中beanshell的用法小结
    LeetCode知识点总结 - 508
    CPU,ISA架构,CPU位数与内存
    计算机毕业设计ssm“逢遇”个人博客平台qaoxv系统+程序+源码+lw+远程部署
    【小程序源码】全新优化版趣味语句生成器
    万界星空科技低代码云MES中的四大现场执行管理模式
    网络技术十二:子网划分
    小程序中wxs
    OpenCV中Mat、Ipllmage以及Halcon中Hobject数据类型之间转换
  • 原文地址:https://blog.csdn.net/run_in_road/article/details/126131960