Swift中的异常和错误处理1

异常处理基础篇

只要我们在编程,就一定要面对错误处理的问题。其实,为了让我们少犯错误,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。

enter image description here
enter image description here

如果我们把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)")
enter image description here
enter image description here

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 六种异常处理的陋习 你觉得自己是一个Java专家吗?是否肯定自己已经全面掌握了Java的异常处理机制?在下面这段代...
    Executing阅读 1,324评论 0 6
  • 转载自:https://github.com/Tim9Liu9/TimLiu-iOS[https://github...
    香橙柚子阅读 8,500评论 0 35
  • 我听歌曲,一般有自己喜欢的歌单,很少跳到外面去,所以李荣浩已经火了很多很多天,到现在我才知道。只第一遍,我就喜欢上...
    一号猫阅读 538评论 1 1
  • 本书讲述的事伊兰特的,五个儿子拥有这上古武器的力量,每个人体内都有神寄居在里面,被称之为寄神者,也只有寄神者才能找...
    涩叙喵阅读 292评论 3 4
  • 2017.08.23号,好热的一天。朋友圈很多人都在晒明星罗志祥来平度的照片,本来也想look一下,到福安市场...
    爱孩子阅读 210评论 0 1