Swift中的异常和错误处理

泊学原文

只要我们在编程,就一定要面对错误处理的问题。其实,为了让我们少犯错误,Swift在设计的时候就尽可能让我们明确感知错误,明确处理错误。例如:

  • 只有使用Optional才能处理空值;

  • switch...case...必须处理所有的请求;

总之,你处处能感受到Swift为你少犯错的良苦用心。所以,当你真的要处理错误的时候,Swift当然更会要求你严谨处理。

如何描述一个错误?

在Swift里,任何一个遵从ErrorType protocol的类型,都可以用于描述错误。ErrorType是一个空的protocol,它唯一的功能,就是告诉Swift编译器,某个类型用来表示一个错误。而通常,我们使用一个enum来定义各种错误。例如,假设我们有一个机器人类型,我们要定一个表达它工作状态的错误:

enum RobotError: ErrorType {
    case LowPower(Double)
    case Overload(Double)
}

其中LowPower表示电量低,它的associated value表示电量的百分比。而Overload表示超过负载,它的associated value表示最大负载值。

如何描述一个会发生错误的方法?

然后,我们来创建一个表示机器人的类:

class Robot {
    var power = 1.0
    let maxLifting = 100.0 // Kg
}

它有两个属性,power表示当前电量,maxLifting表示它可以举起来的最大质量。然后,我们添加一些可以发送给Robot的命令:

enum Command {
    case PowerUp
    case Lifting(Double)
    case Shutdown
}

Command中的三个case分别表示对Robot发送:启动、举重和关机三个命令。

接下来,我们给Robot添加一个接受命令的方法 action。

class Robot {
    var power = 1.0
    let maxLifting = 100.0 // Kg

    func action(command: Command) throws { }
}

由于action有可能发生异常,对于这样的方法,我们要明确使用throws关键字标记它。在action的实现里,我们用一个switch...case来遍历Command:

class Robot {
    var power = 1.0
    let maxLifting = 100.0 // Kg
    
    func action(command: Command) throws {
        switch command {
        case .PowerUp:
            guard self.power > 0.2 else {
                throw RobotError.LowPower(0.2)
            }
            
            print("Robot started")
        case let .Lifting(weight):
            guard weight <= maxLifting else {
                throw RobotError.Overload(maxLifting)
            }
            
            print("Lifting weight: \(weight) KG")
        case .Shutdown:
            print("Robot shuting down...")
        }
    }
}

在action的实现里,当处理.PowerUp命令时,我们使用了guard确保Robot电量要大于20%,否则,我们使用throw RobotError.LowPower(0.2)的方式抛出了一个异常(throw出来的类型必须是ErrorType)。

处理.Lifting命令时,我们读取了.Liftting的associated value,如果要举起的质量大于maxLifting,则throw RobotError.Overload(maxLifting)。

通常,guard和throw配合在一起,可以让我们的代码变的更加简洁。

如何处理错误?

当我们调用了一个可能会抛出异常的方法时,我们一定要"通过某种方式"处理可能会发生的异常,如果你不处理,iOS会替你处理。当然,作为"代劳"的成本,iOS也会Kill掉你的app。因此,对于"业务逻辑类"的异常,我们还是自己处理好些,Swift允许我们使用三种方式处理异常。为了演示它们的用法,我们先来定义一个让Robot工作的函数,由于它会调用action,因此它也会抛出RobotError异常,我们也需要用throws来定义它:

func working(robot: Robot) throws {
}

do...catch...

在working的实现里,首先,我们要让Robot"启动":

func working(robot: Robot) throws {
    do {
        try robot.action(Command.PowerUp)
    }
    catch let RobotError.LowPower(percentage) {
        print("Low power: \(percentage)")
    }
}

通过前面action的代码我们知道,如果传入的robot参数的"电量"低于20%,action会抛出异常,因此在working的实现里:

  • 我们必须在调用会抛出异常的方法前面使用try关键字;

  • 如果我们要捕获方法抛出的异常,就需要把会抛出异常的代码放在关键字do包含的代码块里;

  • 我们使用catch关键字匹配要捕捉的各种异常,例如在上面的例子里,我们捕捉了.LowPower,并且读取了它的associated value;

如果我们要捕获多个异常,就可以在do代码块后面,串联多个catch,例如,我们添加一个让Robot举起某个东西的命令:

func working(robot: Robot) throws {
    do {
        try robot.action(Command.PowerUp)
        try robot.action(Command.Lifting(52))
    }
    catch let RobotError.LowPower(percentage) {
        print("Low power: \(percentage)")
    }
    catch let RobotError.Overload(maxWeight) {
        print("Overloading,max \(maxWeight) KG is allowd")
    }
}

我们就需要在do后面多串联一个catch,用来捕获Robot"超载"的异常。

错、不错都会执行的代码

在Swift的异常处理机制理,有一个允许我们添加无论代码执行正常与否,只要离开当前作用域,就一定会执行的代码。我们使用defer关键字来指定这样的代码。例如,我们给working添加一个defer,它用来让Robot关机。

func working(robot: Robot) throws {
    defer {
        try! robot.action(Command.Shutdown)
    }

    do {
        try robot.action(Command.PowerUp)
        try robot.action(Command.Lifting(52))
    }
    catch let RobotError.LowPower(percentage) {
        print("Low power: \(percentage)")
    }
    catch let RobotError.Overload(maxWeight) {
        print("Overloading,max \(maxWeight) KG is allowd")
    }
}

断言肯定不会错哒~

在上面的defer代码块里,我们使用了"try!"这样的形式。这是由于defer代码块中,不允许我们包含任何会跳出当前代码块的语句,例如:break / return / 抛出异常等。因此,我们使用try!告诉Swift我们确定这个调用不会发生异常(如果你对Swift说谎,是会引发运行时异常的 ^.^)。

另外,使用"try!"标记的函数调用,可以不放在do代码块里。

把错误变成一个Optional

最后,我们调用working函数,让Robot完成工作:

let iRobot = Robot()
try? working(iRobot)

在这里,我们我们使用了"try?"的形式调用了一个会抛出异常的方法,它把表达式的评估结果转换为一个Optional。例如,我们让working返回一个Int:

func working(robot: Robot) throws -> Int {
    defer {
        try! robot.action(Command.Shutdown)
    }

    do {
        try robot.action(Command.PowerUp)
        try robot.action(Command.Lifting(52))
    }
    catch let RobotError.LowPower(percentage) {
        print("Low power: \(percentage)")
    }
    catch let RobotError.Overload(maxWeight) {
        print("Overloading,max \(maxWeight) KG is allowd")
    }

    return 0
}

从上面的代码里可以看到,当函数有返回值的时候,我们要把throws写在返回值前面。

然后,我们查看working的返回值和类型:

let a = try? working(iRobot)
print("value: \(a)\n type: \(a.dynamicType)")

这里,由于我们处理异常,因此a的值是0,但是,a的类型,是一个Optional<Int>。

如果我们把RobotError.Overload注释掉,然后让Robot举起超过100KG的物体:

func working(robot: Robot) throws -> Int {
    defer {
        try! robot.action(Command.Shutdown)
    }

    do {
        try robot.action(Command.PowerUp)
        try robot.action(Command.Lifting(152))
    }
    catch let RobotError.LowPower(percentage) {
        print("Low power: \(percentage)")
    }
    /*catch let RobotError.Overload(maxWeight) {
        print("Overloading,max \(maxWeight) KG is allowd")
    }*/

    return 0
}

这样异常就会被抛到working外围,此时Swift运行时会捕捉到这个异常,并且,把a的值设置成nil:

let a = try? working(iRobot)
print("value: \(a)\n type: \(a.dynamicType)")

接下来?在下一段中,我们将向大家介绍多线程环境中的异常处理。

Swift中的异常和错误处理—— 异常处理基础篇的更多相关文章

  1. ios – 我可以安全地在@try catch块中包装’CoreData无法解决错误’错误

    )是的,我偶尔会得到’CoreData无法完成故障’的错误.在我的特定应用程序中,这通常发生在一种“数据绑定”过程中,因此我可以安全地丢弃故障对象并继续前进.我想通过在@try-catch块中包装数据绑定的循环内部并且只跳过我得到CoreData错误的行来完成此操作.我可以使用CoreData安全地执行此操作吗?

  2. 你如何压缩iOS上的Realm DB?

    我想定期在iOS上压缩一个Realm实例来恢复空间.我认为该过程是将数据库复制到临时位置,然后将其复制回来并使用新的default.realm文件.我的问题是Realm()就像一个单例并且回收对象,所以我无法真正关闭它并告诉它打开新的default.realm文件.这里的文档(https://realm.io/docs/objc/latest/api/Classes/RLMRealm.html)建

  3. ios – 捕获NSKeyedUnarchiver异常

    在Swift中,如果无法取消存档数据,NSKeyedUnarchiver.unarchiveObjectWithData(data)将抛出异常.在某些情况下,我们无法保证数据是否未损坏,例如从文件读取时.我不知道Swift中的try/catch机制,也不知道像canUnarchive这样有助于防止异常的方法.除了在Obj-C中实现try/catch之外,还有一个纯Swift解决方案来解决这个问题吗

  4. ios – 使用swift进行异常处理

    catch来处理它.如果故事板中没有视图控制器,则无法执行任何操作.这是程序员的错误,创建它的人应该处理这些问题.你不能因为这种错误而责怪iOS运行时.

  5. ios – Swift 3 – 将文件夹从主包复制到文档目录

    我的主要包中包含文件夹,我想在首次启动应用程序时将它们复制/剪切到文档目录,以便从那里访问它们.我见过一些例子,但他们都在Obj-C中,我正在使用Swift3.我怎么能这样做?解决方法我设法使用2个功能:

  6. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  7. Swift41/90Days - 面向轨道编程 - Swift 中的异常处理

    问题在开发过程中,异常处理算是比较常见的问题了。我们把下面那根Failure的线路扩展一下,便会看到两条平行的线路,这便是“双轨模型”,这是用“面向轨道编程”思想解决异常处理的理论基础。这就是“面向轨道编程”。也就是说具体的业务只需要处理灰色部分的逻辑:“面向轨道”编程确实给我们提供了一个很有趣的思路。比如ValueTransformation.swift这个真实的完整案例,以及antitypical/Result这个封装完整的Result库。面向铁轨,春暖花开。

  8. 面向轨道编程 - Swift 中的异常处理

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  9. swift详解之十-------------异常处理、类型转换 ( Any and AnyObject )

    异常处理、类型转换注:本文为作者倾心整理,希望对大家有所帮助!在swift中,错误用复合ErrorType协议的值表示。swift处理异常和别的语言不同的是swift不会展开调用堆栈。在swift中throw语句的性能几乎和return一样通过try!所以上面的例子还能这么写结果是一模一样的Any和AnyObject的类型Swift为不确定类型提供了两种特殊类型别名:AnyObject可以代表任何class类型的实例。Any可以表示任何类型,包括方法类型。

  10. Swift 2.0 try? 的替代方法

    你可以配合着if-let或者guard语句来使用try?但你可以试着写出try?实际上我也不太喜欢tryit这个名字,你用你喜欢的名字代替就好。你可以以同样的方法调用它:你仍然不能基于错误类型和错误细节来制定错误处理策略,但是这种实现方式也不像try?你也可以修改tryit函数,让它也能接受做错误处理的代码块,但因为要处理两种不同的代码块,这个函数就会变得相当臃肿。如果要用try而且又需要进行错误处理的话,你就必须得用do-catch或者像结果枚举之类的其它方法了。

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部