swift中defer的实际应用小结

看看苹果官方的介绍

用 defer 语句在即将离开当前代码块时执行一系列语句。该语句让你能执行一些必要的清理工作,不管是以何种方式离开当前代码块的——无论是由于抛出错误而离开,或是由于诸如 return、break 的语句。例如,你可以用 defer 语句来确保文件描述符得以关闭,以及手动分配的内存得以释放。
defer 语句将代码的执行延迟到当前的作用域退出之前。该语句由 defer 关键字和要被延迟执行的语句组成。延迟执行的语句不能包含任何控制转移语句,例如 break、return 语句,或是抛出一个错误。延迟执行的操作会按照它们声明的顺序从后往前执行——也就是说,第一条 defer 语句中的代码最后才执行,第二条 defer 语句中的代码倒数第二个执行,以此类推。最后一条语句会第一个执行。

总结一下苹果官方的介绍

defer语句在代码块(方法、闭包等,可以理解为大括号包装起来的代码)作用域退出之前\color{red}{作用域退出之前}作用域退出之前执行,也就是代码块中其他应该执行的代码都执行完了,才执行defer中的代码
一个代码块允许多个defer,多个defer执行的顺序从后到前\color{red}{从后到前}从后到前

一些测试及误区纠正

测试案例1

func testDefer() {
 defer {
 print("方法中defer内容")
 }
 if true {
 defer {
  print("if 中defer内容")
 }
 print("if中最后的代码")
 }
 print("方法中的代码")
 if true {
 return
 }
 print("方法结束前最后一句代码")
}
testDefer()

以上代码打印结果:

if中最后的代码
if 中defer内容
方法中的代码
方法中defer内容

打印结果中,第一个if中的代码及里面的defer最先执行,方法中的defer最后执行,由此可以看出,代码块中其他能够执行的代码先执行,最后执行defer的内容;defer的作用范围不能简单的看成方法,而是代码块(可能有些同学会有这样的误区)

测试案例2

func testDefer() {
 print("开始")
 defer {
 print("defer 1 中的内容")
 }
 defer {
 print("defer 2 中的内容")
 }
 if true {
 return
 }
 defer {
 print("defer 3 中的内容")
 }
 print("方法结束前最后一句代码")
}
testDefer()

打印结果

开始
defer 2 中的内容
defer 1 中的内容

我们可以看到最后一个defer没有执行,所以defer定义的位置很重要,如果没有执行defer定义的代码,在代码块结束前不会执行defer中的内容

多个defer的执行顺序从后到前

一些实际应用场景

场景1:一些资源用完后需释放,这里给的是官方的一个案例

func processFile(filename: String) throws {
 if exists(filename) {
 let file = open(filename)
 defer {
  close(file)
 }
 while let line = try file.readline() {
  // 处理文件。
 }
 // close(file) 会在这里被调用,即作用域的最后。
 }
}

开始用到资源的时候就使用defer去释放,避免忘记释放资源

场景2:加锁解锁,借鉴了kingfisher

let lock = NSLock()
func testDefer() {
 lock.lock()
 defer {
 lock.unlock()
 }

 doSomething()
}
testDefer()

在加锁后立刻用defer解锁,避免忘记解锁

场景3:处理一些代码块作用域结束前的重复操作,比如请求网络数据的时候

通常的一种写法

func loadCityList(_ finish: ((Error?, [String]?) -> ())?) {
 DispatchQueue.global().async { // 模拟网络请求
 let data: AnyObject? // 模拟服务器返回的数据
 guard let dict = data as? [String: AnyObject] else {
  DispatchQueue.main.async {
  finish?(error, nil)
  }
  return
 }
 guard let code = dict["code"] as? Int, code == 200 else {
  DispatchQueue.main.async {
  finish?(error, nil)
  }
  return
 }
 guard let citys = dict["data"] as? [String]? else {
  DispatchQueue.main.async {
  finish?(error, nil)
  }
  return
 }
 DispatchQueue.main.async {
  finish?(nil, citys)
 }
 }
}

当每次有错误处理时和结果正确时都需要去做回调,而且回调可能有一堆代码,看起来代码会比较冗余,而且在一些错误处理时很容易造成忘记回调

defer怎么去写呢

func loadCityList(_ finish: ((Error?, [String]?) -> ())?) {
 DispatchQueue.global().async { // 模拟网络请求
 var error: Error? = nil
 var citys: [String]? = nil
 defer {
  DispatchQueue.main.async {
  finish?(error, citys)
  }
 }

 let data: AnyObject? // 模拟服务器返回的数据
 guard let dict = data as? [String: AnyObject] else {
  error = ...
  return
 }
 guard let code = dict["code"] as? Int, code == 200 else {
  error = ...
  return
 }
 guard let tempCitys = dict["data"] as? [String]? else {
  error = ...
  return
 }
 citys = tempCitys
 }
}

使用defer既解决了代码冗余,又解决了可能忘记回调的问题,还有当我们看到defer时,我们很清楚知道,无论网络请求结果如果,都会回调

总结

本文主要介绍了defer的定义、作用及一些用法

到此这篇关于swift中defer的实际应用的文章就介绍到这了,更多相关swift中defer应用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Swift中defer关键字推迟执行示例详解

    前言 大家应该都知道,在一些语言中,有try/finally这样的控制语句,比如Java. 这种语句可以让我们在finally代码块中执行必须要执行的代码,不管之前怎样的兴风作浪. 在Swift 2.0中,Apple提供了defer关键字,让我们可以实现同样的效果. func checkSomething() { print("CheckPoint 1") doSomething() print("CheckPoint 4") } func doSomething(

  • swift中defer几个简单的使用场景详解

    前言 最近准备把 swift 文档再扫一遍,发现了 defer 这个关键字,defer 是个非常重要的 swift 语言特征,恕本人愚钝,以前还从来没有用过这个呢~ 简单地列一下这个东西有哪些可以用得上的情景吧~~话不多说了,来一起看看详细的介绍吧. defer 是干什么用的 很简单,用一句话概括,就是 defer block 里的代码会在函数 return 之前执行,无论函数是从哪个分支 return 的,还是有 throw,还是自然而然走到最后一行. 这个关键字就跟 Java 里的 try-

  • Swift中defer的正确使用方法

    defer 是干什么用的 很简单,用一句话概括,就是 defer block 里的代码会在函数 return 之前执行,无论函数是从哪个分支 return 的,还是有 throw,还是自然而然走到最后一行. 这个关键字就跟 Java 里的 try-catch-finally 的finally一样,不管 try catch 走哪个分支,它都会在函数 return 之前执行.而且它比 Java 的finally还更强大的一点是,它可以独立于 try catch 存在,所以它也可以成为整理函数流程的一

  • swift中defer的实际应用小结

    看看苹果官方的介绍 用 defer 语句在即将离开当前代码块时执行一系列语句.该语句让你能执行一些必要的清理工作,不管是以何种方式离开当前代码块的--无论是由于抛出错误而离开,或是由于诸如 return.break 的语句.例如,你可以用 defer 语句来确保文件描述符得以关闭,以及手动分配的内存得以释放. defer 语句将代码的执行延迟到当前的作用域退出之前.该语句由 defer 关键字和要被延迟执行的语句组成.延迟执行的语句不能包含任何控制转移语句,例如 break.return 语句,

  • Swift中初始化init的方法小结

    前言 我们在深入初始化方法之前,不妨先再想想Swift中的初始化想要达到一种怎样的目的. 其实就是安全.在Objective-C中,init方法是非常不安全的:没有人能保证init只被调用一次,也没有人保证在初始化方法调用以后,实例的各个变量都完成初始化,甚至如果在初始化里使用属性进行设置的话,还可能会造成各种问题.虽然Apple也明确说明了不应该在init中使用属性来访问,但这并不是编译器强制的,因此还是会有很多开发者犯这样的错误. 所以Swift有了超级严格的初始化方法.一方面,Swift强

  • swift中的正则表达式小结

    作为一门先进的编程语言,Swift 可以说吸收了众多其他先进语言的优点,但是有一点却是让人略微失望的,就是 Swift 至今为止并没有在语言层面上支持正则表达式. 正则表达式的用处: 判断给定的字符串是否符合某一种规则(专门用于操作字符串) - 电话号码,电子邮箱,URL... - 可以直接百度别人写好的正则 - 别人真的写好了,而且测试过了,我们可以直接用 - 要写出没有漏洞正则判断,需要大量的测试,通常最终结果非常负责 过滤筛选字符串,网络爬虫 替换文字,QQ聊天,图文混排 语法规则 使用过

  • Swift中的指针操作详解

    前言 Objective-C和C语言经常需要使用到指针.Swift中的数据类型由于良好的设计,使其可以和基于指针的C语言API无缝混用.但是语法上有很大的差别. 默认情况下,Swift 是内存安全的,这意味着它禁止我们直接操作内存,并且确保所有的变量在使用前都已经被正确地初始化了.但是,Swift 也提供了我们使用指针直接操作内存的方法,直接操作内存是很危险的行为,很容易就出现错误,因此官方将直接操作内存称为 "unsafe 特性". 一旦我们开始直接操作内存,一切就得靠我们自己了,因

  • Swift 中的 JSON 反序列化示例详解

    目录 业界常用的几种方案 手动解码方案,如 Unbox(DEPRECATED) 阿里开源的 HandyJSON 基于 Sourcery 的元编程方案 Swift build-in API Codable 属性装饰器,如 BetterCodable 各个方案优缺点对比 Codable 介绍 原理浅析 Decoder.Container 协议 自研方案 功能设计 Decoder.Container 具体实现 再议 PropertyWrapper 应用场景示例 单元测试 性能对比 业界常用的几种方案

  • Swift中通知中心(NotificationCenter)的使用示例

    前言 本文主要介绍了关于Swift通知中心(NotificationCenter)使用的相关内容,NotificationCenter是Swift中一个调度消息通知的类,采用单例模式设计,实现传值.回调等作用. 通知的作用还是挺强大的,对于两个不相关的控制器之间,要进行信息的传递,使用通知是个不错的选择,下面话不多说了,来一起看看详细的使用方法吧. 1.添加通知 /// 通知名 let notificationName = "XMNotification" /// 自定义通知 Noti

  • Swift中动态调用实例方法介绍

    在 Swift 中有一类很有意思的写法,可以让我们不直接使用实例来调用这个实例上的方法,而是通过类型取出这个类型的某个实例方法的签名,然后再通过传递实例来拿到实际需要调用的方法.比如我们有这样的定义: 复制代码 代码如下: class MyClass {     func method(number: Int) -> Int {         return number + 1     } } 想要调用 method 方法的话,最普通的使用方式是生成MyClass的实例,然后用.method来

随机推荐