您的位置:首页 > 移动开发 > Swift

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语句捕获并处理,这样就导致了这个错误传播给该函数或方法的调用者,如果它的调用者也都不捕获处理,那么最后程序会出现运行期错误。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: