摘要: swift的异常错误抛出以及其处理,使你的代码更加健壮,更加安全
tips:新手开发一枚,如有一些概念错误,还请指正。原创文章,转载请注明出处。
2015年的wwdc大会之前,swift还没有异常错误处理。这是个惹人非议的诟病。然而苹果公司不负众望,在swift2.0上发布了该项更新。下面我们来了解下try catch组合的异常处理机制
建立专属的错误类型
在oc中,我们通常使用NSError来做对应的错误处理,在swift中,要想使用try catch这套机制,错误类型必须满足ErrorType协议,NSError也是符合这个协议的。枚举是作为错误展示的比较合适的数据结构。
下面我们来针对数组是否为空使用是否越界为例子,来声明一个错误类型:
enum myError : ErrorType{
case NOCOUNT
case OVERCOUNT
}
在开发中,可以根据具体环境需要,建立自己需要的错误类型。
抛出错误的函数
有了错误类型之后,那就是在合适的时机去抛出这些错误,如何抛出错误呢?我们需要对我们的函数进行一些简单的改造。
/// 判断数组使用是否安全
/// - parameter - myArray 使用的数组
/// - userCount使用的index
/// - returns: 该数组传入位置的值
fun cuseArraySafe(myArray : NSArray,useCount : Int)throws->Int{
if myArray.count<1{
print("array is empty!")
throw myError.NOCOUNT
}else if(myArray.count>0&& myArray.count> useCount){
print("use safe")
return myArray[useCount]as!Int
}else{
print("use over count")
throw myError.OVERCOUNT
}
}
在参数之后,返回值->之前,我们加入了throws这个关键字,这样我们的函数就可以使用throw抛出错误了。只需要在何时的判断之后,抛出我们枚举的错误,就完成了我们的基础工作。
机关已经设置好了,夹子已经放好了,就等着猎物上钩了。让我们开始使用这套机关吧。
使用抛出错误的函数
声明有throws的函数必须使用do..try去实现:
let arr :NSArray=NSArray.init(array: [1,2,3,4,5])
do{
let num =try useArraySafe(arr, useCount:6)
print("\(num)");
}catch myError.NOCOUNT{
print("array is empty!")
}catc hmyError.OVERCOUNT{
print("use over count")
}
do内部是对函数的实现过程,在实现过程中,如果抛出了错误,则会在do后面的catch中根据抛出错误的类型进行处理,且要使用try去调用函数。此处例子抛出错误使用的是if语句,降低了代码的可读性。如果使用guard语句的话,可以很自然地在内部使用throw抛出错误,且代码可读性高。下面是guard的改写
/// 判断数组使用是否安全
/// - parameter - myArray 使用的数组
/// - userCount使用的index
/// - returns: 该数组传入位置的值
fun cuseArraySafe(myArray : NSArray,useCount : Int)throws->Int{
defer{
print("all done!")
}
guard(myArray.count>0&& myArray.count> useCount)else{
throw myError.USELESS
}
returnmyArray[useCount]as!Int
}
let arr :NSArray=NSArray.init(array: [1,2,3,4,5])
do{
let num = tryuseArraySafe(arr, useCount:6)
print("\(num)");
}catch myError.USELESS{
print("wrong")
}
在myError内新增一个USELESS, 当且仅当数组不为空和数组不越界的时候才能使用,否则错误。
可以看到,我在原函数内部添加了一个defer。defer的作用是,在它所在作用域代码执行完毕,它负责擦屁股的,做一些最后的操作。所以defer所处作用域的确认十分重要。
不关心错误类型,只关心出没出错
有的时候,我们并不关心到底出什么错误,我们仅仅关心到底出没出错。这时我们在do内调用函数的时候,仅仅需要在try后面加一个?就行了,如下:
do{
let num =try? useArraySafe(arr, useCount:6)
print("\(num)");
}catch myError.USELESS{
print("wrong")
}
这个时候,会提示你后面的catch语句将不会被执行,我们就可以忽略错误类型,只在没出错的情况下调用函数成功。使用try?还会将函数的返回值自动转化为optional类型。void返回值的函数try?后返回的是optional.None也就是nil。如果返回的事int类型的100,try?后返回的就是optional(100)。
不处理错误异常
当你十分有信心你的操作不会抛出错误,不处理错误异常,你仅需要在try调用函数的时候加一个!就行了。但是一旦你的程序出现了错误或者异常,那么使用try!的后果就是你的程序会crash,所以小心使用。
总结:使用ErrorType来声明你的错误类型,在函数中使用throws来告诉编译器,我这个函数调用有异常错误抛出,使用throw抛出你的错误。在调用时使用do..try..catch的组合完成错误异常处理。try?和try!应该在合适的时候使用。