Swift语法 -- [18 - 错误处理]

2020-10-29  本文已影响0人  happy神悦

开发过程常见的错误

1.自定义错误

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

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

 var result = try divide(20, 10)

2.do-catch

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

3.处理Error

处理Error的2种方式
1 通过do-catch捕捉Error
2 不捕捉Error,在当前函数增加throws声明,Error将自动抛给上层函数

 func test() throws {
    print("1")
    print(try divide(20, 0))
    print("2") 
}
try test()

 do {
    print(try divide(20, 0))
} catch is SomeError {
    print("SomeError")
}

4.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()

a、b是等价的

var a = try? divide(20, 0)
var b: Int?
do {
    b = try divide(20, 0)
} catch { b = nil }

5.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)

6.defer

func open(_ filename: String) -> Int { 
     print("open")
     return 0 
}
func close(_ file: Int) {
    print("close")
}

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

func fn1() { print("fn1") }
func fn2() { print("fn2") }
func test() {
    defer { fn1() }
    defer { fn2() }
}
test()
// fn2
// fn1

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

7.assert(断言)

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

8. 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()

上一篇 下一篇

猜你喜欢

热点阅读