美文网首页
swift开发之try.. catch的理解与使用(MD)

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

作者: littlewish | 来源:发表于2017-03-04 16:53 被阅读210次
摘要: 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!应该在合适的时候使用。

老文地址

相关文章

网友评论

      本文标题:swift开发之try.. catch的理解与使用(MD)

      本文链接:https://www.haomeiwen.com/subject/carjgttx.html