Swift错误处理

参考:

  • Magical Error Handling in Swift
  • Play-with-Swift-2/21-Error-Handling

强制退出程序

使用assert,表达式为假,程序强制退出

assert(1>0,"Error")

assert还有一种形式为assertionFailure,程序走到这个函数时,程序会直接终端,并打印中断的信息

assertionFailure("failure")

注意assertassertionFailure都只在开发的过程中起作用,当release程序后,assertassertionFailure就自动失效了。

不过有些时候,希望程序在真正执行起来的时候,在某些情况下,遇到一些严重的错误,程序要强制退出,可以使用preconditionfatalError

precondition(1>0,"Error") fatalError("Error")

Error

在一般的处理过程中可以返回nil表示错误,但是只返回nil,不能表示出错的原因。比如连接网络,如果网络连接不上,就返回一个nil,这样做也是可以的,但是此时只是知道网络没连接上而已。至于是服务端出错还是客户端出错,就不得而知了。所以需要一套完善的错误处理机制。

1.自定义错误类型:

  • 遵守Error协议

如下:

enum vendingError: Error{
        case NoSuchItem
        case NotEnoughMoney(Int)
        case OutOfStock
    }

2.抛出错误,使用throws关键字,抛出错误后,整个函数就已经结束了,所以后面就不用return
真正抛出异常使用throw关键字

func vend(itemName itemName: String,money: Int) throws -> Int{

        guard let item = items[itemName] else{
            throw vendingMachine.vendingError.NoSuchItem
        }

        guard money >= item.price else{
            throw vendingMachine.vendingError.NotEnoughMoney(item.price)
        }

        guard item.count > 0 else{
            throw vendingMachine.vendingError.OutOfStock
        }

        return money - item.price
    }
   }

此时,这个函数就有两种退出的方式:

1.正确的return
2.抛出异常

错误处理

对于抛出异常的函数,就不能直接调用了,要处理异常。使用try关键字

try!表示坚信函数不会抛出异常,但是这样写也是有风险的。

try! machine.vend(itemName: "Coca Cola",money: pocketMoney)

try?表示如果抛出异常,返回值为nil。所以对于返回值可以使用解包的形式:

if let leftMoney = try? machine.vend(itemName: "Coca Cola",money: pocketMoney)
{

}else{

}

如果希望更具错误的类型,来做不同的处理。就需要使用docatch关键字,如下:

do{
    pocketMoney = try machine.vend(itemName: "Coca Cola",money: pocketMoney)
    print(pocketMoney,"Yuan left")
}catch{
    print("Error occured during vending")
}

要处理不同类型的异常,形式如下:

do{
    pocketMoney = try machine.vend(itemName: "Coca Cola",money: pocketMoney)
    print(pocketMoney,"Yuan left")
}catch vendingMachine.vendingError.NoSuchItem{
    print("No Such Item")
}catch vendingMachine.vendingError.NotEnoughMoney(let price){
    print("No Enough Money",price,"Yuan needed.")
}catch vendingMachine.vendingError.OutOfStock{
    print("Out of Stock")
}catch{
    print("Error occured during vending.")
}

还有一种形式是:

do{
    pocketMoney = try machine.vend(itemName: "Coca Cola","Yuan left")
}catch let error as vendingMachine.vendingError{

}catch{
    print("Error occured during vending.")
}

defer

在其它语言中,错误处理通常有个finally关键字,不管程序是否抛出异常,finally中的代码都要执行。

swift中引入了defer,它应该写在有可能退出这个函数的语句之前。defer本身的意识是延迟,表示延迟执行。

func vend(itemName itemName: String,money: Int) throws -> Int{

        defer {
            print("Hava a nice day")
        }

        guard let item = items[itemName] else{
            throw vendingMachine.vendingError.NoSuchItem
        }
        ......
    }

如果有多个defer,是按倒序来执行的,如下的方法:

func vend(itemName itemName: String,money: Int) throws -> Int{

        defer {
            print("Hava a nice day")
        }

        guard let item = items[itemName] else{
            throw vendingMachine.vendingError.NoSuchItem
        }

        guard money >= item.price else{
            throw vendingMachine.vendingError.NotEnoughMoney(item.price)
        }

        guard item.count > 0 else{
            throw vendingMachine.vendingError.OutOfStock
        }

        defer {
            print("Thank you")
        }

        dispenseItem(itemName: itemName)

        return money - item.price
    }

    private func dispenseItem(itemName: String){
        items[itemName]!.count -= 1
        print("Enjoy your",itemName)
    }

如果调用如下的方法,正常的执行,不会抛出异常:

var pocketMoney = 3
try? machine.vend(itemName: "Coca Cola",money: pocketMoney)

则其输出顺序为:

Enjoy your Coca Cola
Thank you
Hava a nice day

如果调用如下方法,抛出异常:

var pocketMoney = 2
try? machine.vend(itemName: "Coca Cola",money: pocketMoney)

则控制台只会输出:

Hava a nice day

这是因为抛出异常结束程序,只会找抛出异常之前的defer语句

Swift错误处理的更多相关文章

  1. ios – 无法识别的选择器发送到实例NSTimer Swift

    解决方法让updateTime成为一个类方法.如果它是在一个纯粹的Swift类中,你需要在@objc前面说明该方法的声明,如:

  2. 如何在iOS中为预期的assert / assertionFailure编写单元测试?

    这里的问题是,当someString参数为空字符串时,您可以保证函数不会失败–在您的实际应用程序中.这是因为断言不在发布版本中运行.结果是你可以在开发过程中使用assert作为调试的一种形式,但如果这种情况在现实生活中发生,你应该按顺序处理它,而不是崩溃.因此测试断言“发生”是否真的不是一种有效的单元测试技术,这就是为什么你以这种方式使用它的麻烦.

  3. ios – 类型推断(自动类型检测)如何在swift中工作?

    LLVM如何检测变量是一个字符串?

  4. ios – Swift可选项:语言问题,还是做错了什么?

    应该有可选的类型;type是但是,如果我这样做,它的工作原理:它似乎是基本的替代,但我可能会遗漏一些语言的细微差别.谁能对此有所了解?之后就像暧昧一样,更多,这是我的解决方案:这适用于所有非对象Swift对象,包括Swift字符串,数字等.感谢Viktor提醒我String不是Swift中的对象.如果您知道值的类型,您可以替换任何?使用适当的可选类型,如String?

  5. ios – 覆盖Swift中的超类委托

    我正在开发一个包含两个UIViews的Swift(v1.2)项目.MyView和MyViewSubclass.MyView有一个委托,我想在MyViewSubclass中覆盖它作为一个子协议,类似于UITableViews有一个UITableViewDelegate,它也符合超级uiscrollviewdelegate.我的第一个想法是覆盖超类属性,但这会导致编译器错误,因为子类不能覆盖具有不同类

  6. ios – 我可以在swift中将字符串转换为代码块吗?

    有没有办法将字符串转换为代码块?

  7. ios – Swift:方法重载只在返回类型上有所不同

    我一直在看Swift类,其中定义了两种方法,它们的返回类型不同.我不习惯使用允许这种语言的语言,所以我去寻找描述它如何在Swift中工作的文档.我在任何地方都找不到任何东西.我本来期望在Swift书中有关于它的整个部分.这记录在哪里?

  8. ios – 字符串资源Xcode swift

    我是iOS开发和Swift语言的新功能.而且我尝试制作简单的iOS应用程序,我需要在应用程序中使用一些字符串资源.当然,我可以将这个字符串放在我的*.swift文件中作为常量,但我认为这是一个坏的方法.我该怎么做?

  9. ios – 如何使用新的Apple Swift语言发布JSON

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

  10. ios – 核心数据并发调试:假阳性

    如WWDC2014session225(核心数据新增内容)所述,iOS8和OSX优胜美地的核心数据现在支持命令行参数-com.apple.CoreData.ConcurrencyDebug1,以启用检测到违反CoreData的并发合同的断言.在我的实验中,我发现它在iOS8beta1(在设备和模拟器中)都可以工作,但是我似乎发现了一个错误的肯定,即框架是抛出一个多线程违规异常不这样做至少这是我的信

随机推荐

  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,所以编译器会报错,现在来一一解决。

返回
顶部