swift开发之try.. catch的理解与使用

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: 该数组传入位置的值
func useArraySafe(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!")
}catch myError.OVERCOUNT{
    print("use over count")
}



    do内部是对函数的实现过程,在实现过程中,如果抛出了错误,则会在do后面的catch中根据抛出错误的类型进行处理,且要使用try去调用函数。此处例子抛出错误使用的是if语句,降低了代码的可读性。如果使用guard语句的话,可以很自然地在内部使用throw抛出错误,且代码可读性高。下面是guard的改写

/// 判断数组使用是否安全
/// - parameter - myArray 使用的数组
///             - userCount使用的index
/// - returns: 该数组传入位置的值
func useArraySafe(myArray : NSArray,useCount : Int) throws -> Int{
    defer{
        print("all done!")
    }
    guard (myArray.count > 0 && myArray.count > useCount) else{
        throw myError.USELESS
    }
    return myArray[useCount] as! Int
}

let arr : NSArray = NSArray.init(array: [1,2,3,4,5])

do{
    let num = try useArraySafe(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!应该在合适的时候使用。


    

你可能感兴趣的:(异常处理,swift,try,try!,try?)