一.异常的介绍
只要我们在编程,就一定要面对错误处理的问题。
Swift在设计的时候就尽可能让我们明确感知错误,明确处理错误
比如:只有使用Optional才能处理空值;
如何描述一个错误?
在Swift里,任何一个遵从ErrorType protocol的类型,都可以用于描述错误。
ErrorType是一个空的protocol,它唯一的功能,就是告诉Swift编译器,某个类型用来表示一个错误。
通常,我们使用一个enum来定义各种错误的可能性
二.异常的示例
假如我们想要读取一个文件中的内容,按照OC的逻辑我们可以这样来模拟
当我们调用方法获取结果为nil时,你并不能确定到底参数了什么错误得到了nil
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("abc")
- 使用异常对上述方法进行改进
// 1.定义异常
enum FileReadError : ErrorType {
case FileISNull
case FileNotFound
}
// 2.改进方法,让方法抛出异常
func readFileContent(filePath : String) throws -> String {
// 1.filePath为""
if filePath == "" {
throw FileReadError.FileISNull
}
// 2.filepath有值,但是没有对应的文件
if filePath != "/User/Desktop/123.plist" {
throw FileReadError.FileISNull
}
// 3.取出其中的内容
return "123"
}
三.处理异常有三种方式
3.异常的处理三种方式
3.1.try方式,需要手动处理异常
do {
let result = try readFileContent("abc")
} catch {
print(error)
}
3.2.try?方式,不处理异常,如果出现了异常,则返回一个nil.没有异常,则返回对应的值
最终返回结果为一个可选类型
let result = try? readFileContent("abc")
3.3.try!方法,告诉系统该方法没有异常.
注意:如果出现了异常,则程序会崩溃
try! readFileContent("abc")
如何抛出异常
在抛出异常之前,我们需要在函数或方法的返回箭头 -> 前使用 throws 来标明将会抛出异常
func myMethodRetrunString() throws -> String
// No return, we can just add throws in the end
func myMethodRetrunNothing() throws
声明之后, 我们需要在函数或者方法里扔出异常,很简单使用throw 就可以了
func myMethod() throws
//...
// item is an optional value
guard let item = item else {
// need throws the error out
throw MyError.NotExist
}
// do with item
}
上面这段代码使用了guard
来进行unwrap optional value。这是 Swift 2.0 提供的一个新的方法。
Guard
在 Haskell, Erlang 等语言中早已存在guard
, 在这里有更多关于它的介绍。
guard
翻译过来可以理解为守护,守卫。
在 Swift 中,guard
有点像if
但是他们有两个非常重要的区别
guard
必须强制有else
语句
只有在guard
审查的条件成立,guard
之后的代码才会运行 (像守卫一样,条件不符就不让过去)。
guard
中的else
只能执行转换语句,像return
,break
,continue
或者throws
当然你也可以在这里返后一个函数或者方法。
值得注意的是,guard
的使用会提高你代码的可读性,但是也代表你的代码的执行会有非常明确的顺序性,这一点需要开发者们留心处理。
虽然我们在异常处理中提到了guard
但是不代表它只能在异常处理中使用。它具有广泛的适用性,或许过阵子我会专门为guard
的使用写篇文章。
如何获取并处理异常?
上文讲述了如何建造抛出异常,获取和处理异常就变得很简单了。使用do-catch
机制。
文/阮超(简书作者)原文链接://www.greatytc.com/p/96a7db3fde00著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
do {
try functionWillThrowError()
} catch {
// deal with error
}
do-catch
机制简单易懂。很多编程语言也使用类似的机制进行异常处理,但是在 Swift 中有一个比较重要的特性。
catch
和switch
一样具有 Pattern Matching 的能力。所以,使用catch
你可以对异常的解析进行更为高级的处理
文/阮超(简书作者)原文链接://www.greatytc.com/p/96a7db3fde00著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
do {
try functionWillThrowError()
} catch MyError.NotExist {
// deal with not exist
} catch MyError.OutOfRange {
// deal with not exist
}
这里值得提一下在 Swift 2.0中一个跟异常处理没有关系的改进
Swift 2.0 中没有了 do-while循环,取而代之的是 repeat-while。苹果说这个改动是为了增强代码的可读性。但是我更觉得是为了让我们更舒服的使用 do-catch
不处理异常
如果我不想处理异常怎么办,或者说,我非常确定某个方法或者函数虽然声明会抛出异常,但是我自己知道我在使用时候是绝对不会抛出任何异常的。这种情况下 我们可以使用 try!
文/阮超(简书作者)
原文链接://www.greatytc.com/p/96a7db3fde00
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
try! functionThrowErrorNil()
当然,如果你使用 try!,而你的方法或者函数抛出了异常,那么你会得到一个运行中异常 (runtime error) 所以我们开发者需要慎用哦。
Defer
文章结束前我们再讨论下 defer
在你的代码块就要结束前。如果你使用了 defer。 在其之中的代码就会运行。等于说,给了你最后的机会来进行一些处理。如果你熟悉 BDD 或者 TDD, 那么你可以参考他们中的 aferAll 机制
文/阮超(简书作者)
原文链接://www.greatytc.com/p/96a7db3fde00
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
func myFunction() throws {
defer {
// No matter what happened I need do something
print("All done, clean up here")
}
guard let item = item else {
// need throws the error out
throw MyError.NotExist
}
guard item.count > maxNumber else {
// need throws the error out
throw MyError.OutOfRange
}
// do something with item
// ...
}
注意,如果你有多个defer 语句,他们在执行的顺序会和栈一样,最后一个进,第一个出.
总结
使用 ErrorType 的帮助建立你的异常类型
使用 throws 来声明异常,用 throw 来抛出异常
使用 do-catch 机制来获取和处理异常