美文网首页SwiftTips程序员iOS Developer
iOS开发之带你畅游闭包Closure --Swift

iOS开发之带你畅游闭包Closure --Swift

作者: Qinz | 来源:发表于2017-03-17 10:55 被阅读1305次
Qinz
在Swift中引进了闭包Closure的概念,使用起来更加的方便和简洁了,下面让我们揭开它的神秘面纱,带你畅行它执行的详细过程,熟悉理解本文中的每个实例,你就对闭包有了深刻认识了,废话不多说,开始畅游~
一:首先我们创建一个Cat的类,声明一个name的常量和description的变量
class Cat{
    let name:String
    init(name:String) {
        self.name = name
    }
    var description:String{return "<名字 \(name)>"}
    
    deinit {
        print("🐱对象被销毁")
    }
}

1.写一个延迟执行的闭包函数,这里延迟主要是方便我们接下来的观察

 
    func delay(seconds:Int,closure:@escaping ()->()){
        
        print("当前时间为\(Date())")
        
        DispatchQueue(label: "com.ys").asyncAfter(deadline: .now() + .seconds(seconds), execute: {
            
        print("延迟执行所在的线程\(Thread.current)")
            
            print("当前时间为\(Date())")
            
            print("🐱")
            
            closure()
        })
    }

2.我们来调用闭包函数

   func demo1(){
        let pokemon1 = Cat(name: "ys")
        print("-----开始执行demo1的函数-----\(pokemon1)")
        
        delay(seconds: 2, closure: {

            print("-----执行闭包中-------\(pokemon1)")
        })
        print("当前所在的线程\(Thread.current)")

        print("--------demo1的函数执行结束---------------------")
    }

3.我们来看看控制台输出了什么

-----开始执行demo1的函数-----ClosureSamples.Cat
当前时间为2017-03-17 02:10:33 +0000
--------demo1的函数执行结束---------------------
当前时间为2017-03-17 02:10:35 +0000
🐱
-----执行闭包中-------ClosureSamples.Cat

🐱对象被销毁

4.分析如下

  • a:首先我们调用了demo1()方法,它输出”-----开始执行demo1函数------“的log,这个毋容置疑。

  • b:接下来demo1()里面又调用了delay(seconds:Int,closure:@escaping ()->())这个含有闭包参数的函数,说明下,这里有一个关键字escaping表示的是逃逸闭包,就是闭包中的变量可以被捕获在调用的地方引用,这里我们研究的是闭包执行的过程,所以并没有给闭包参数。

  • c:执行delay()这个函数,打印出了当前的时间,我们只看秒数,因为这里我们是延迟2秒执行 现在的秒数是33。

  • d:现在打印出”----demo1函数执行结束------“,估计在这里有些人容易懵逼,可能会想为什么不先执行DispatchQueue里面的代码打印出35秒这个当前时间呢?原因很简单,因为打印"----demo1函数执行结束------"代码在主线程中执行,而延迟执行的代码是在子线程执行的,如果你觉得还不理解,建议先看看这篇多线程的基础知识详解,点击这里跳转

  • e:对于步骤d理解后,接下来我们再来看,打印出35秒的时间,说明延迟2秒代码开始执行了,然后我们在DispatchQueue中开始执行闭包closure(),这时候代码跳转至demo1()函数的delay中开始执行闭包中的代码,打印出“-----执行闭包中-------”,我们可以看到此时这只还是活着的,当闭包代码执行完毕,接着就走了Cat类中的deinit
    方法,这时就挂了。

5.我们来打印出当前的线程


    func demo1(){
        let pokemon1 = Cat(name: "ys")
        print("-----开始执行demo1的函数-----\(pokemon1)")
        
        delay(seconds: 0, closure: {

            print("-----执行闭包中-------\(pokemon1)")
        })
        print("当前所在的线程\(Thread.current)")

        print("--------demo1的函数执行结束---------------------")
    }
    
    
    func delay(seconds:Int,closure:@escaping ()->()){
        
        print("当前时间为\(Date())")
        
        DispatchQueue(label: "com.ys").asyncAfter(deadline: .now() + .seconds(seconds), execute: {
            
        print("延迟执行所在的线程\(Thread.current)")
            
            print("当前时间为\(Date())")
            
            print("🐱")
            
            closure()
        })
    }

6.控制台输出为

-----开始执行demo1的函数-----ClosureSamples.Cat
当前时间为2017-03-17 02:16:25 +0000
--------demo1的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x610000066bc0>{number = 3, name = (null)}
当前时间为2017-03-17 02:16:25 +0000
🐱
-----执行闭包中-------ClosureSamples.Cat
🐱对象被销毁

  • 总结

对于上面闭包执行过程的总结:demo1() 函数执行完毕后,闭包才开始执行;并且2秒后当闭包被执行的时候 实例依然存活着。这是因为闭包捕获(强引用)了 变量:编译器发现在闭包内部引用了 变量,它会自动捕获该变量(默认是强引用),所以 的生命周期与闭包自身是一致的。因此,闭包有点像精灵球 ,只要你持有着精灵球闭包, 变量也就会在那里,不过一旦精灵球闭包被释放,引用的 也会被释放。例子中:一旦 GCD 执行完毕,闭包就会被释放,所以 的 deinit 方法也会被调用。值得注意的是 Swift 在闭包执行时才会取出捕获变量的值[^1],所以这里的性能消耗是很小的,我们可以认为它之前捕获的是变量的引用(或指针)。

二:接下来我们在demo1的基础上加上对变量进行赋值的代码,看第二个例子
   func demo2(){
    
        var pokemon2 = Cat(name: "王子🍡")
        print("-----开始执行demo2的函数-----\(pokemon2.name)")
        delay(seconds: 2, closure: {
            
            print("-----执行闭包中-------\(pokemon2.name)")
        
        })
        pokemon2 = Cat(name: "公主👸")
        
        print("--------demo2的函数执行结束---------------------")
   
    }

1.我们看看控制台输出的内容

-----开始执行demo2的函数-----王子🍡
当前时间为2017-03-17 02:22:18 +0000
🐱对象被销毁
--------demo2的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x618000264ac0>{number = 3, name = (null)}
当前时间为2017-03-17 02:22:20 +0000
🐱
-----执行闭包中-------公主👸
🐱对象被销毁

2.分析

  • a 对于前两段输出内容肯定毋容置疑。

  • b 为什么到第三段直接输出了"对象被销毁"呢?是不是又有点懵逼了,不慌--带你看是什么原理,此处和demo1()不同的是我们将pokemon2由原来的“王子”重新赋值给了“公主”,此时引用发生了变化,所以“王子”顺利挂彩,“公主”接上,那么接下来主线程中的"--------demo2的函数执行结束---------------------"开始输出,接着延迟2秒至32,开始打印DispatchQueue中的代码,然后执行闭包,开始跳转至demo2()中的delay函数执行闭包里面的代码,打印出“-----执行闭包中-------公主”,我们可以看到此时“公主””还是活着的,当闭包代码执行完毕,接着就走了Cat类中的deinit方法,这时“公主”也挂了!

  • 总结

对于二的总结:在创建完闭包之后修改了 pokemon 对象,闭包延迟2秒后执行(虽然此时已经脱离了 demo2() 函数的作用域),我们打印的结果是新的 pokemon 对象,而不是旧的!这是因为 Swift 默认捕获的是变量的引用:首先初始化一个值为 "王子" 的 pokemon 对象,接着修改该对象的值为 "公主",之前值为 "王子" 的对象由于没有其他变量强引用,所以会被释放。接着闭包等待2秒钟执行,打印捕获 "公主" 变量(引用)的内容,待闭包执行完毕“公主”也就被释放了。

三:我们来看一个值类型的闭包捕获过程
  func demo3() {
        var value = 200
        print("-----开始执行demo3的函数-----\(value)")
        
        delay(seconds: 1, closure: {
            
            value = 10000
            print("-----执行闭包中-------\(value)")
        })
        
        value = 230
        print("--------demo3的函数执行结束---------------------")
    }

1.我们来看看控制台输出了什么

-----开始执行demo3的函数-----200
当前时间为2017-03-17 02:24:55 +0000
--------demo3的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x6100000701c0>{number = 3, name = (null)}
当前时间为2017-03-17 02:24:56 +0000
🐱
-----执行闭包中-------10000

  • 总结

  • 理解了demo2也就理解demo3了,可以看出值类型和字符串类型执行的是一样的,所以针对于demo2的特性:对于值类型也是一样的,闭包打印了新的整型变量值——尽管整型变量是值类型!因为它捕获了变量的引用,而不是变量自身的内容!,如果捕获的是变量 var(而不是常量 let),你也可以在闭包中[^2]修改它的值

四:我们在demo3的基础上做一点点小小的改动
    func demo4(){
    
        var value = 100
        
        print("-----开始执行demo4的函数-----\(value)")

        delay(seconds: 1, closure: { [oldValue = value] in
          
            
            print("-----执行闭包中-------\(oldValue)")

            })
        value = 800
        
        print("--------demo4的函数执行结束---------------------")
    }
  1. 我们来看看控制台输出了什么:很有趣是不是吗?你会发现最后输出的是100,这就是闭包一个很好的特征,可以拿到之前的值,我们在这里写了[oldValue = value]就可以拿到oldValue之前的值,也就是被赋值为800之前的值,此时的100是旧值,800是新的值。

-----开始执行demo4的函数-----100
当前时间为2017-03-17 02:29:26 +0000
--------demo4的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x618000074a40>{number = 3, name = (null)}
当前时间为2017-03-17 02:29:27 +0000
🐱
-----执行闭包中-------100

2.如果你想,800到哪里去了呢?没错,它就是此时的value,看打印输出

  
    func demo4(){
    
        var value = 100
        
        print("-----开始执行demo4的函数-----\(value)")

        delay(seconds: 1, closure: { [oldValue = value] in
          
            
            print("-----执行闭包中-------\(oldValue)")
            print("-----执行闭包中-------\(value)")
            
        })
        value = 800
        
        print("--------demo4的函数执行结束---------------------")
    }

-----开始执行demo4的函数-----100
当前时间为2017-03-17 02:30:51 +0000
--------demo4的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x60800006f0c0>{number = 3, name = (null)}
当前时间为2017-03-17 02:30:52 +0000
🐱
-----执行闭包中-------100
-----执行闭包中-------800

3.有人会想我能不能对oldValue再在闭包里面赋值呢?孩子你想多了---显然它是一个let类型啊!

五:根据以上,我们来看个稍微复杂的示例
    func demo5() {
        var pokemon = Cat(name: "王子 🍡")
        print("-----开始执行demo4的函数-----\(pokemon.name)")
        delay(seconds: 1, closure: { [pokemonCopy = pokemon] in
            
            print("-----执行闭包中-------\(pokemonCopy.name)")
            print("******执行闭包中*******\(pokemon.name)")
            
        })
        pokemon = Cat(name: "公主 👸")
        print("--------demo5的函数执行结束---------------------")
    }

1.我们来看看控制台的输出:经过上面熟悉了四个例子的前提,是不是很容易明白下面的输出结果了呢?如果还不明白,看我下面的分析。有点懵逼的就是为什么”王子 “被赋值为"公主 "之后"王子 "没有被销毁,原因就在于在闭包的参数中我们写了[pokemonCopy = pokemon],这样我们又对旧值有了引用,不会消失了吧?我们看看有这句和没有这句的区别 ,下面是对比:

-----开始执行demo4的函数-----王子 🍡
当前时间为2017-03-17 02:34:41 +0000
--------demo5的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x60000007a480>{number = 3, name = (null)}
当前时间为2017-03-17 02:34:42 +0000
🐱
-----执行闭包中-------王子 🍡
******执行闭包中*******公主 👸
🐱对象被销毁
🐱对象被销毁

不写[pokemonCopy = pokemon]

    func demo5() {
        var pokemon = Cat(name: "王子 🍡")
        print("-----开始执行demo4的函数-----\(pokemon.name)")
        delay(seconds: 1, closure: {
            
//            print("-----执行闭包中-------\(pokemonCopy.name)")
            print("******执行闭包中*******\(pokemon.name)")
            
        })
        pokemon = Cat(name: "公主 👸")
        print("--------demo5的函数执行结束---------------------")
    }

控制台输出

-----开始执行demo4的函数-----王子 🍡
当前时间为2017-03-17 02:36:07 +0000
🐱对象被销毁
--------demo5的函数执行结束---------------------
延迟执行所在的线程<NSThread: 0x610000260a80>{number = 3, name = (null)}
当前时间为2017-03-17 02:36:08 +0000
🐱
******执行闭包中*******公主 👸
🐱对象被销毁

分析

  • a ”王子 “被创造。

  • b 接着闭包捕获了 ”王子 “ 的拷贝(这里实际上是捕获了 pokemon 变量的值)。 所以当我们紧接着为 pokemon 变量赋新值 “公主 ” 后,“王子 ” 还没有被释放,依然被闭包所保留。

  • c 当我们离开 demo5 函数的作用域,”公主 “ 就被释放了,在方法内部 pokemon 变量自身只被一个强引用所保持,离开作用域强引用也就消失了。

  • d 稍后闭包执行时,打印了 "王子 “,这是因为在闭包创建时,捕获列表就捕获了 Pokemon。

  • e 最后 GCD 释放了闭包,由此可以证明闭包保持了"王子 “的引用。

六:好了,最后来看一个综合例子

1.先想想下面这段代码会怎么输出?

func demo6() {
    var pokemon = Cat(name: "王子 🍡")
    print("----- 初始化 pokemon 为 \(pokemon.name)")
    
    delay(seconds: 2, closure:  { [capturedPokemon = pokemon] in
        print("closure 1 — 旧值: \(capturedPokemon.name)")
        print("closure 1 — 新值: \(pokemon.name)")
        pokemon = Cat(name: "公主 👸")
        print("closure 1 - pokemon的值为 \(pokemon.name)")
    })
    
    pokemon = Cat(name: "爱心 ❤️")
    print("****** pokemon 发生改变为 \(pokemon.name)")
    
    delay(seconds: 2, closure:  { [capturedPokemon = pokemon] in
        print("closure 2 — 旧值: \(capturedPokemon.name)")
        print("closure 2 — 新值: \(pokemon.name)")
        pokemon = Cat(name: "青蛙 🐸")
        print("closure 2 - pokemon的值为 \(pokemon.name)")
    })
}

2.来~我们看看控制台输出结果:估计你看到又懵逼了,不急,我们来慢慢分析为什么是这样输出的,如果你理解了这个,说明闭包就真的已经深入理解了。

----- 初始化 pokemon 为 王子 🍡
当前时间为2017-03-17 02:41:00 +0000
****** pokemon 发生改变为 爱心 ❤️
当前时间为2017-03-17 02:41:00 +0000
延迟执行所在的线程<NSThread: 0x6000002624c0>{number = 4, name = (null)}
延迟执行所在的线程<NSThread: 0x61800007ca00>{number = 3, name = (null)}
当前时间为2017-03-17 02:41:02 +0000
当前时间为2017-03-17 02:41:02 +0000
🐱
closure 2 — 旧值: 爱心 ❤️
🐱
closure 1 — 新值: 爱心 ❤️
closure 1 — 旧值: 王子 🍡
closure 2 - pokemon的值为 青蛙 🐸
closure 1 - 新值: 青蛙 🐸
🐱对象被销毁
🐱对象被销毁
closure 1 - pokemon的值为 公主 👸

🐱对象被销毁
🐱对象被销毁

分析

  • 1:输出”----- 初始化 pokemon 为 王子 “肯定是毋庸置疑的。
  • 2:输出”当前时间为2017-03-16 08:43:40 +0000“也肯定完全理解。
  • 3:输出”****** pokemon 发生改变为 爱心 ❤️“原因是这段输出是在主线程中执行,不懂点击这里先熟悉多线程知识
  • 4:输出”当前时间为2017-03-16 08:43:40 +0000“说明开始进入第一次调用的delay方法中。
  • 5:输出”延迟执行所在的线程{number = 3, name = (null)}“说明进入DispatchQueue开启子线程执行代码
  • 6:为什么在第5步输出”延迟执行所在的线程{number = 3, name = (null)}“接着打印”延迟执行所在的线程{number = 5, name = (null)}“,因为此时进入到了我们第二次调用delay的方法中,”number = 3“和”number = 5“可以看出两次调用delay的方法是在不同的子线程中进行的异步并发任务,这点不明白,还是点击这里先熟悉多线程知识
  • 7:输出”当前时间为2017-03-16 08:43:42 +0000“肯定也是毋庸置疑的。
  • 8:接着又输出”当前时间为2017-03-16 08:43:42 +0000“,其实原理和5,6步是一样的,也就是它们执行的是并行任务。
  • 9:接下来打印了一只”“,毋容置疑
  • 10:这里开始懵逼?按照5,6和7,8的步骤规律,应该接下来再打印一只猫才对,怎么乱入了”closure 2 — 旧值: 爱心 ❤️“呢?其实这个还是并行任务的特点,我们再次运行一遍程序,发现输出结果如下:两只”“又同时输出了,其实发现它们就是打印一遍delay第一次调用的方法里面的代码,再打印一次delay第二次调用的方法里面的代码,交替进行,也没先后的。

----- 初始化 pokemon 为 王子 🍡
当前时间为2017-03-17 02:43:39 +0000
****** pokemon 发生改变为 爱心 ❤️
当前时间为2017-03-17 02:43:39 +0000
延迟执行所在的线程<NSThread: 0x608000071a80>{number = 3, name = (null)}
延迟执行所在的线程<NSThread: 0x610000073b80>{number = 4, name = (null)}
当前时间为2017-03-17 02:43:41 +0000
当前时间为2017-03-17 02:43:41 +0000
🐱
🐱
closure 1 — 旧值: 王子 🍡
closure 2 — 旧值: 爱心 ❤️
closure 1 — 新值: 爱心 ❤️
closure 2 — 新值: 爱心 ❤️
closure 1 - pokemon的值为 公主 👸
closure 2 - pokemon的值为 青蛙 🐸
🐱对象被销毁
🐱对象被销毁
🐱对象被销毁

🐱对象被销毁

  • 11:接下来执行了closure()代码,回到delay的闭包中,一直从”closure 1 — 旧值: 王子 “到输出”closure 2 - pokemon的值为 青蛙 “都是交替进行,各自输出一次的。

  • 12:为什么最后四只”“是一起挂的,因为我们在闭包里面都对旧值有了引用,也就是和第五个步骤的原理是一样的,所以都是等闭包代码执行完才被释放。

总结:Swift中的Closure还是很方便和有趣的,了解和熟悉了它执行的过程对于以后的开发非常有利,因为你会发现在OC的工程中我们使用了大量的Block,而Swift把它变得更加优雅了。

我是Qinz,希望我的文章对你有帮助。

相关文章

  • iOS开发之带你畅游闭包Closure --Swift

    在Swift中引进了闭包Closure的概念,使用起来更加的方便和简洁了,下面让我们揭开它的神秘面纱,带你畅行它执...

  • Swift5.0 - day4-闭包、属性、方法、下标

    一、闭包 1.1、闭包表达式(Closure Expression)在 Swift 里面可以通过函数 func 定...

  • Swift 5.3 - SE-0279 Multiple Tra

    在最初 Swift 的定义中,当方法的最后一个参数为闭包时,称该闭包为尾随闭包(trailing closure)...

  • JavaScript----闭包

    javascript之闭包 闭包的概念     闭包(closure)是 JavaScript 的一种语法特性。 ...

  • Swift closure闭包

    究竟什么是Closure? 说的通俗一点,一个函数加上它捕获的变量一起,才算一个closure //MARK: -...

  • swift:Closure 闭包

    @noescape @autoclosure @autoclosure(escaping)

  • Swift Closure闭包

    函数也是一种闭包 console log 如下 闭包表达式 console log 如下 闭包中的参数类型推断 c...

  • swift 闭包(closure)

    闭包:就是一个函数和它所捕获的变量/常量环境组合起来,称为闭包。1.一般定义在函数内部的函数。2.一般它捕获的是外...

  • JavaScript学习之路-闭包

    一、闭包? 闭包一词想必iOS开发的童鞋指定很熟悉,Objective-C上的闭包叫Block,Swift上就叫闭...

  • Swift 闭包

    @Author Swift 闭包(Closure) 闭包是一种可以在代码中作为参数传递,自含的功能块。 闭包类似于...

网友评论

    本文标题:iOS开发之带你畅游闭包Closure --Swift

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