欢迎使用CSDN-markdown编辑器

来源:互联网 发布:apache下载安装 win7 编辑:程序博客网 时间:2024/06/14 20:00

前言

关于我们为什么要使用异常处理,请看百度百科为我们作出的描述,想要更详细的资料请点这里

异常处理,英文名为exceptional handling, 是代替日渐衰落的error code方法的新法,提供error code 所未能具体的优势。异常处理分离了接收和处理错误代码。这个功能理清了编程者的思绪,也帮助代码增强了可读性,方便了维护者的阅读和理解。 异常处理(又称为错误处理)功能提供了处理程序运行时出现的任何意外或异常情况的方法。异常处理使用 try、catch 和 finally 关键字来尝试可能未成功的操作,处理失败,以及在事后清理资源。
异常处理,是编程语言或计算机硬件里的一种机制,用于处理软件或信息系统中出现的异常状况(即超出程序正常执行流程的某些特殊条件)。

它有功能强大的控制流语句,像dotrycatchguarddefer。我们能够使用这些关键字去操控我们的代码,更精确地执行我们的代码。
以上摘自百度百科:

关联,在Objective-C中,异常处理一般都是使用NSError类接收异常和抛出异常,使用方法像这样

NSError *error = nil;
NSString *string = [[NSString alloc] initWithContentsOfFile:@”filePath” encoding:NSUTF8StringEncoding error:&error];
if (error != nil) {
// Exception handle
NSLog(@”Erorr is %@”, error);
}
// Code…
不得不说,Swift的异常处理更为优雅,下面会重点介绍。

Demo

你可以在github上下载这个示例Demo

构建异常类型

我们可以构建自己的异常类型,它只是一个枚举,遵循ErrorType,像是这样

enum MyErrorHandling: ErrorType {

case error1case error2

}
定义函数可抛出异常

这里值得一提的是,只有函数才能够抛出异常。我们需要在参数后和返回值前加上throws关键字,并用guard捕捉异常,用throw抛出异常。

func myLoad(item: String?) throws -> String {
guard let newItem = item else {
throw MyErrorHandling.error1
}
return newItem
}
guard: guard必须与else配合使用,只有当guard审查的条件成立,guard之后的代码才会执行,否则抛出异常。
使用guard还有一个好处,guard定义的变量,在guard后面也可以使用,并且是已经解包了,newItem就是这样。
捕获/处理异常

我们使用do-try-catch去获取并处理异常

do {
try myLoad(nil)
} catch {
print(“error”)
}
如果只包含一个catch语句,那么所有的错误都会在这个catch中执行,我们能够捕抓其错误信息

do {
try myLoad(nil)
} catch let error as NSError {
print(error)
}
我们在使用catch时,它是能够进行模式匹配的,我们能够进行更精准的错误匹配处理

do {
try myLoad(“test”)
} catch MyErrorHandling.error1 {
print(“error1”)
} catch MyErrorHandling.error2 {
print(“error2”)
}
如果我们不需要捕抓错误,那么我们可以使用try!或try?去执行该函数

try? myLoad(nil)
不建议使用try!,使用try?会更加安全。因为如果当有错误捕抓到时,程序会直接崩溃
Defer(延迟执行)

那么我们何时需要使用到defer?这里使用官方的例子,比如我们需要读取某文件内容并处理,你需要打开这个文件,最后你需要关闭这个文件。当一切顺利的时候,程序一直往下运行,文件也将会被关闭。那么当中间出现一些错误呢?例如在某个环节执行失败了,但是你依然是需要去关闭该文件,在此defer就表现得很强大了。在这个例子中,我们可以把关闭文件的代码放到defer里面,在defer里的代码无论是函数执行成功或失败都会被执行。

func myLoad(item: String?) throws -> String {
defer {
print(“Handle 1”)
}

defer {    print("Handle 2")}guard let newItem = item else {    throw MyErrorHandling.error1}return newItem

}
需要注意的是,defer语句可以有多个,它们的执行顺序是Handle 1 -> Handle 2。不难理解,它们的执行顺序是先进后出的。
这里再说一下,你可以在github上下载这个示例Demo

普通函数使用Guard
我们可以在普通的函数里使用guard,即不带抛异常的函数。可以用来代替if去使用。

我编写了一个printlnName的函数,它接受一个参数name。内部只有一个判断,如果内部参数name为空时,打印错误

func printlnName(name: String) {
guard name != “” else {
print(“Name should be not null”)
return
}
}

printlnName(“”)

0 0