错误处理
错误处理是响应错误以及从错误中返回的过程。swift提供第一类错误支持,包括在运行时抛出,捕获,传送和控制可回收错误。
一些函数和方法不能总保证能够执行所有代码或产生有用的输出。可空类型用来表示值可能为空,但是当函数执行失败的事后,可空通常可以用来确定执行失败的原因,因此代码可以正确地响应失败。在Swift中,这叫做抛出函数或者抛出方法。
错误的表示
在Swift中,错误用符合ErrorType
协议的值表示。 Swift枚举特别适合把一系列相关的错误组合在一起,同时可以把一些相关的值和错误关联在一起。因此编译器会为实现ErrorType
协议的Swift枚举类型自动实现相应合成。
// 模拟自动售卖机的错误情况
enum VendingMachineError: ErrorType {
// 商品不存在时
case InvalidSelection
// 投入的钱不够时
case InsufficientFunds(required: Double)
// 商品卖完了
case OutOfStock
}
错误抛出 通过在函数或方法声明的参数后面加上throws
关键字,表明这个函数或方法可以抛出错误。如果指定一个返回值,可以把throws
关键字放在返回箭头(->)的前面。
func canThrowError() throw -> String func cannotThrowError() -> String
当调用一个抛出函数的时候,在调用前面加上try
。这个关键字表明函数可以抛出错误。
let favoriteSnacks = [
"Alice": "Chips",
"Bob": "Licorice",
"Eve": "Pretzels",
] func buyFavoriteSnack(person: String) throws {
let snackName = favoriteSnacks[person] ?? "Candy Bar"
try vend(itemName: snackName)
}
捕捉和处理错误
使用do-catch语句来就捕获和处理错误
do {
try vend(itemName: "Candy Bar")
// 如果vend会抛出错误的话,会传递到catch中
// 可以在catch里处理发生错误后应做的事情,不会造成运行时错误
} catch VendingMachineError.InvalidSelection {
print("Invalid Selection")
} catch VendingMachineError.InsufficientFunds(let amountRequired) {
print("Insufficient funds. Please insert an additional $\(amountRequired).")
} catch VendingMachineError.OutOfStock {
print("Out of Stock")
}
// Insufficient funds. Please insert an additional $0.25.
注意:Swift中的错误处理和其他语言中的异常处理很像,使用了try
、catch
和throw
关键字。但是和这些语言——包括Objective-C——不同的是,Swift不会展开调用堆栈,那会带来很大的性能损耗。因此,在Swift中throw
语句的性能可以做到几乎和return
语句一样。
禁止错误传播
在运行时,有几种情况抛出函数事实上是不会抛出错误的。在这几种情况下,你可以用forced-try
表达式来调用抛出函数或方法,即使用try!
来代替try
。
通过try!
来调用抛出函数或方法禁止了错误传送,并且把调用包装在运行时断言,这样就不会抛出错误。如果错误真的抛出了,会触发运行时错误。
enum CustomError: ErrorType {
case SomeError
} func willOnlyThrowIfTrue(value: Bool) throws {
if value { throw CustomError.SomeError }
} do {
try willOnlyThrowIfTrue(true)
} catch {
print("Error")
} try! willOnlyThrowIfTrue(true) // 运行时错误
收尾操作
使用defer语句来在执行一系列的语句。这样不管有没有错误发生,都可以执行一些必要的收尾操作。包括关闭打开的文件描述符以及释放所有手动分配的内存。
defer
语句把执行推迟到退出当前域的时候。defer
语句包括defer
关键字以及后面要执行的语句。被推迟的语句可能不包含任何将执行流程转移到外部的代码,比如break
或者return
语句,或者通过抛出一个错误。被推迟的操作的执行的顺序和他们定义的顺序相反,也就是说,在第一个defer
语句中的代码在第二个defer
语句中的代码之后执行。
func processFile(filename: String) throws {
if exists(filename) {
let file = open(filename)
defer {
close(file)
}
while let line = try file.readline() {
// 写文件
}
// defer中的close(file) 会在最后调用。不管有没有抛出错误
}
}