Swift—do-try-catch错误处理模式-备
2016-05-03 11:10
447 查看
Swift 1.x的错误处理模式存在很多弊端,例如:为了在编程时候省事,给error参数传递一个nil,或者方法调用完成后不去判断error是否为nil,不进行错误处理。
let contents = NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding, error: nil)//error参数传递一个nil
或者
var err: NSError?
let contents = NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding, error: &err)
不好的编程习惯,由于Objective-C和Swift 1.x没有强制处理机制,因此一旦真的发生错误,程序就会发生崩溃。
同样的从文件中读取字符串示例,如果使用Swift2错误处理模式代码如下:
import Foundation
do { //要做一些操作
let str = try NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding) //要尝试做的事情
} catch let err as NSError { //如果失败则进入catch代码块
err.description
}
do-try-catch这种错误模式与Java中异常处理机制非常类似,本意就是尝试(try)做一件事情,如果失败则捕获(catch)处理。
捕获错误
完整的do-try-catch错误处理模式的语法如下:
do {
try 语句
成功处理语句组
} catch 匹配错误 {
错误处理语句组
}
在try 语句中可以产生错误,当然也可能不会产生错误,如果有错误发生,catch就会处理错误。catch代码块可以有多个,错误由哪个catch代码块处理是由catch后面的错误匹配与否而定的。错误类型的多少就决定了catch可以有多少。我们先介绍一下错误类型。
错误类型
在Swift中错误类型必须遵从ErrorType协议,其次考虑到错误类型的匹配,它应该被设计成为枚举类型,枚举类型非常适合将一组相关值关联起来。
如果我们编写访问数据库表程序,实现对表数据插入、删除、修改和查询等操作,我们会需要类似如下代码的错误类型:
enum DAOError: ErrorType {
case NoData
case PrimaryKeyNull
}
NoData表示没有数据情况,PrimaryKeyNull表示表的主键(Primary Key)为空情况。
那么我们就可以通过如下代码捕获错误。
do {
//try 访问数据表函数或方法
} catch DAOError.NoData {
print("没有数据。")
} catch DAOError.PrimaryKeyNull {
print("主键为空。")
}
能放到try后面调用函数或方法都是有要求的,他们是有可能抛出错误,在这些函数或方法声明的参数后面要加上throws关键字,表示这个函数或方法可以抛出错误。
声明抛出错误方法示例代码如下:
[html] view plain copy
print?
//删除Note记录方法
func remove(model: Note) throws {
...
}
//查询所有记录数据方法
func findAll() throws -> [Note] {
...
}
上述代码remove(_:)方法没有返回值,throws关键字放到参数后面。findAll()有返回值throws关键字放到参数和返回值类型之间。
在函数或方法中抛出错误
一个函数或方法能够声明抛出错误,是因为在函数或方法中产生并抛出了错误,这样函数或方法声明抛出错误才有实际的意义。
在产生并抛出错误方式:
在函数或方法中通过throw语句,人为地抛出错误。
在函数或方法中调用的其他可以抛出错误函数或方法,但是没有捕获处理,会导致错误被传播出来。
示例代码如下:
[html] view plain copy
print?
//删除Note方法
func remove(model: Note) throws {
guard let date = model.date else { //判断抛出时候是有了guard语句
//抛出"主键为空"错误
throw DAOError.PrimaryKeyNull
}
//比较日期主键是否相等
for (index, note) in listData.enumerate() where note.date == date {
listData.removeAtIndex(index)
}
}
//查询所有数据方法
func findAll() throws -> [Note] {
guard listData.count > 0 else { //判断抛出时候是有了guard语句
//抛出"没有数据"错误。
throw DAOError.NoData
}
return listData
}
func printNotes() throws { //声明抛出错误
let datas = try findAll()
for note in datas {
print("date : \(note.date!) - content: \(note.content!)")
}
}
try printNotes()
guard语句最擅长处理这种早期判断,条件为false情况下抛出错误。
findAll()语句本身有可能产生错误,但是并没有使用catch语句捕获并处理,这样就导致了这个错误传播给该函数或方法的调用者,如果它的调用者也都不捕获处理,那么最后程序会出现运行期错误。
let contents = NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding, error: nil)//error参数传递一个nil
或者
var err: NSError?
let contents = NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding, error: &err)
不好的编程习惯,由于Objective-C和Swift 1.x没有强制处理机制,因此一旦真的发生错误,程序就会发生崩溃。
同样的从文件中读取字符串示例,如果使用Swift2错误处理模式代码如下:
import Foundation
do { //要做一些操作
let str = try NSString(contentsOfFile: filePath,
Êencoding: NSUTF8StringEncoding) //要尝试做的事情
} catch let err as NSError { //如果失败则进入catch代码块
err.description
}
do-try-catch这种错误模式与Java中异常处理机制非常类似,本意就是尝试(try)做一件事情,如果失败则捕获(catch)处理。
捕获错误
完整的do-try-catch错误处理模式的语法如下:
do {
try 语句
成功处理语句组
} catch 匹配错误 {
错误处理语句组
}
在try 语句中可以产生错误,当然也可能不会产生错误,如果有错误发生,catch就会处理错误。catch代码块可以有多个,错误由哪个catch代码块处理是由catch后面的错误匹配与否而定的。错误类型的多少就决定了catch可以有多少。我们先介绍一下错误类型。
错误类型
在Swift中错误类型必须遵从ErrorType协议,其次考虑到错误类型的匹配,它应该被设计成为枚举类型,枚举类型非常适合将一组相关值关联起来。
如果我们编写访问数据库表程序,实现对表数据插入、删除、修改和查询等操作,我们会需要类似如下代码的错误类型:
enum DAOError: ErrorType {
case NoData
case PrimaryKeyNull
}
NoData表示没有数据情况,PrimaryKeyNull表示表的主键(Primary Key)为空情况。
那么我们就可以通过如下代码捕获错误。
do {
//try 访问数据表函数或方法
} catch DAOError.NoData {
print("没有数据。")
} catch DAOError.PrimaryKeyNull {
print("主键为空。")
}
能放到try后面调用函数或方法都是有要求的,他们是有可能抛出错误,在这些函数或方法声明的参数后面要加上throws关键字,表示这个函数或方法可以抛出错误。
声明抛出错误方法示例代码如下:
[html] view plain copy
print?
//删除Note记录方法
func remove(model: Note) throws {
...
}
//查询所有记录数据方法
func findAll() throws -> [Note] {
...
}
上述代码remove(_:)方法没有返回值,throws关键字放到参数后面。findAll()有返回值throws关键字放到参数和返回值类型之间。
在函数或方法中抛出错误
一个函数或方法能够声明抛出错误,是因为在函数或方法中产生并抛出了错误,这样函数或方法声明抛出错误才有实际的意义。
在产生并抛出错误方式:
在函数或方法中通过throw语句,人为地抛出错误。
在函数或方法中调用的其他可以抛出错误函数或方法,但是没有捕获处理,会导致错误被传播出来。
示例代码如下:
[html] view plain copy
print?
//删除Note方法
func remove(model: Note) throws {
guard let date = model.date else { //判断抛出时候是有了guard语句
//抛出"主键为空"错误
throw DAOError.PrimaryKeyNull
}
//比较日期主键是否相等
for (index, note) in listData.enumerate() where note.date == date {
listData.removeAtIndex(index)
}
}
//查询所有数据方法
func findAll() throws -> [Note] {
guard listData.count > 0 else { //判断抛出时候是有了guard语句
//抛出"没有数据"错误。
throw DAOError.NoData
}
return listData
}
func printNotes() throws { //声明抛出错误
let datas = try findAll()
for note in datas {
print("date : \(note.date!) - content: \(note.content!)")
}
}
try printNotes()
guard语句最擅长处理这种早期判断,条件为false情况下抛出错误。
findAll()语句本身有可能产生错误,但是并没有使用catch语句捕获并处理,这样就导致了这个错误传播给该函数或方法的调用者,如果它的调用者也都不捕获处理,那么最后程序会出现运行期错误。
相关文章推荐