ios 开发

swift错误处理

2022-12-10  本文已影响0人  iOS小洁

错误类型

语法错误(编译报错)

逻辑错误

运行时错误(可能会导致闪退,一般也叫做异常)

...

自定义错误

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

enum SomeError : Error { 
    case illegalArg(String) 
    case outOfBounds(Int, Int) 
    case outOfMemory 
}

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

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

可以使用do-catch捕捉Error

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

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

var result = try divide(20, 10)

do { 
    try divide(20, 0) 
} catch let error { 
    switch error { 
        case let SomeError.illegalArg(msg):
            print("参数错误:", msg) 
        default:
            print("其他错误")
  }
}

处理error

处理Error的2种方式

try?、try!

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

func test() {
    print("1") 
    var result1 = try? divide(20, 10) // Optional(2), Int? 
    var result2 = try? divide(20, 0) // nil 
    var result3 = try! divide(20, 10) // 2, Int 
    print("2")
}
test()

// 以下ab是等价的
var a = try? divide(20, 0) 
var b: Int?
do { 
    b = try divide(20, 0) 
} catch { 
    b = nil 
}

rethrows

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

func exec(_ fn: (Int, Int) throws -> Int, _ num1: Int, _ num2: Int) rethrows { 
    print(try fn(num1, num2)) 
}
// Fatal error: Error raised at top level 
try exec(divide, 20, 0)

defer

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

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

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

func processFile(_ filename: String) throws {
    let file = open(filename) 
  defer { close(file) } 
  // 使用file 
  // ....
    try divide(20, 0)
    // close将会在这里调用
} try 
processFile("test.txt")


func fn1() { print("fn1") } 
func fn2() { print("fn2") } 
func test() {
    defer { fn1() }
    defer { fn2() } 
} 
test()  // 先执行fn2,后执行fn1

assert(断言)

很多编程语言都有断言机制:不符合指定条件就抛出运行时错误,常用于调试(Debug)阶段的条件判断

默认情况下,Swift的断言只会在Debug模式下生效,Release模式下会忽略

增加Swift Flags修改断言的默认行为

func divide(_ v1: Int, _ v2: Int) -> Int { 
    assert(v2 != 0, "除数不能为0") 
    return v1 / v2 
} 
print(divide(20, 0))

fatalError

如果遇到严重问题,希望结束程序运行时,可以直接使用fatalError函数抛出错误(这是无法通过do-catch捕捉的错误)

使用了fatalError函数,就不需要再写return

在某些不得不实现、但不希望别人调用的方法,可以考虑内部使用fatalError函数

func test(_ num: Int) -> Int { 
    if num >= 0 { return 1 } 
  fatalError("num不能小于0") 
}


class Person { 
  required init() {} 
} 
class Student : Person {
    required init() { 
    fatalError("don't call Student.init") 
    }
    init(score: Int) {} 
} 
var stu1 = Student(score: 98) 
var stu2 = Student()

局部作用域

可以使用 do 实现局部作用域

do {
    let dog1 = Dog() 
    dog1.age = 10 
    dog1.run()
}
上一篇下一篇

猜你喜欢

热点阅读