认识Swift系列7之闭包

2019-07-11  本文已影响0人  Niuszeng
    // 1.闭包表达式 Closure expression
    func test_closure_expression() {
        // 闭包的定义
        /**
         用in来分开闭包类型和语句块代码
         {
            (参数...)->返回值类型 in
            闭包代码
         }
         */
        
        
        // 定义一个函数
        func sum(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
        // 定义一个闭包表达式
        var closure = {
            (v1: Int, v2: Int)->Int in return v1 + v2
        }
        
        // 注意:虽然定义闭包的时候写了参数标签,但是调用的时候不写标签
//        print("Closure expression \(sum(1, 2))")
//        print("Closure expression \(closure(1, 2))")
        
        // 匿名闭包
//        {
//            (v1: Int, v2: Int)->Void in
//            print("v1 + v2 = \(v1 + v2)")
//        }(1, 2)
        
        func exec(fn:(Int, Int)->Int){}
        exec { _ , _ in 10} // 当不需要用到参数时,可以使用_忽略
    }
    test_closure_expression()
    
    // 2.闭包表达式的简化
    func test_closure_simplify() {
        func exec(v1: Int, v2: Int, sum: (Int, Int)->Int) {
            print("closure_simplify->exec:\(sum(v1, v2))")
        }
        
        // 正常调用
        exec(v1: 1, v2: 2, sum: { (x1: Int, x2: Int) -> Int in
            return x1 + x2
        })
        
        // 注意:以下一些列省略系编译器自动推断技术
        // 简化1
        // (由于exec函数已经明确需要两个Int类型参数,因此此处可以省略参数类型和小括号)
        //  (由于exec函数已经明确返回Int类型的参数,因此可省略返回值类型)
        exec(v1: 1, v2: 2, sum: { x1, x2 in
            return x1 + x2
        })
        
        // 简化2(单条语句,简化掉return)
        exec(v1: 1, v2: 2, sum: { x1, x2 in x1 + x2 })
        
        // 简化3(可以省略参数列表和in)
        // 使用美元符$+数字表示第几个参数
        exec(v1: 1, v2: 2, sum: { $0 + $1 })
        
        // 简化4-1(只有两个参数,直接写个+号,编译器会自动推断出是加法运算)
        exec(v1: 1, v2: 2, sum: + ) // 两个参数才可以
        
        // 简化4-2(尾随闭包)
        exec(v1: 1, v2: 2) {$0 + $1}
        
        
        // Test ...
        func exec1(v1: Int, v2: Int, v3: Int, sum: (Int, Int, Int)->Int) {
            print("closure_simplify->exec:\(sum(v1, v2, v3))")
        }
        exec1(v1: 1, v2: 2, v3: 3, sum: {$0 + $1 + $2})
        exec1(v1: 1, v2: 2, v3: 3) {$0 + $1 + $2}
    }
    test_closure_simplify()
    
    // 3.尾随闭包
    func test_tail_closure() {
        // 将闭包作为函数的最后一个参数,可以将该闭包写到小括号外,类似函数体,增强可读性
        func exec(v1: Int, v2: Int, v3: Int, sum: (Int, Int, Int)->Int) {
            print("test_tail_closure->exec:\(sum(v1, v2, v3))")
        }
        exec(v1: 1, v2: 2, v3: 3, sum: {$0 + $1 + $2} )
        exec(v1: 1, v2: 2, v3: 3) { $0 + $1 + $2 } // 尾随
        
        func exec1(fn: (Int, Int, Int)->Int) {
            print("test_tail_closure->exec1:\(fn(1,2,3))")
        }
        exec1(fn: {$0 + $1 + $2}) // 正常调用
        exec1() {$0 + $1 + $2} // 尾随1
        exec1 {$0 + $1 + $2} // 尾随2,省略小括号
    }
    test_tail_closure()
    
    // 4.闭包值捕获
    func test_closure_capturing_values() {
        // 将闭包作为函数的最后一个参数,可以将该闭包写到小括号外,类似函数体,增强可读性
        typealias Fn = (Int)->Int
        func getFn()->Fn {
            var num = 0
            func plus(_ i: Int)->Int {
                num += i
                return num
            }
            return plus
        }
        
        let fn = getFn()
        fn(4) // 4
        fn(1) // 5
        
        /** 注意:这里fn其实就是内部的plus
         由于num实际上是getFn栈空间的一个局部变量,当 let fn = getFn() 完成后,getFn栈空间会销毁
         但是fn使用num的时候,getFn栈空间已经销毁,因此为了保存num的值,fn(plus)会将num的值捕获
         为了要捕获num的值,闭包的实现方式将从函数升级为对象
         
         这里的底层实现实际上是在堆空间创建了一个 fn对象,fn的内存结构如下
            fn堆空间内存
         ┏━━━━━━━━━━━━━━━━━━━━━━┓
         ┃   classInfo          ┃
         ┃  retainCount         ┃
         ┃    var num           ┃
         ┃  plus(Int)->Int{...} ┃
         ┗━━━━━━━━━━━━━━━━━━━━━━┛
         
         即fn拥有一个成员变量 num 和 一个方法plus,即fn底层实现是使用了对象来实现
         
         当fn中没有捕获外层函数栈空间的局部变量时,不会发生值捕获
         当fn内部需要操作的值是全局变量时,也不会发生值捕获
         fn发生值捕获的唯一条件就是要捕获的值声明周期过期的情况(一半在函数栈空间的局部变量)
         */
    }
    test_closure_capturing_values()
    
    // 5.自动闭包
    func test_auto_closure() {
        // 一个需求:如果第一个数乘5大于20,返回第一个,否则返回第二个乘5
        
        // 使用如下函数实现
        func getFirstMoreThan20(_ v1: Int, _ v2: Int)->Int {
            return v1 > 20 ? v1 : v2
        }
        
        let a = 5
        let b = 6
        getFirstMoreThan20(a * 5, b * 5)
        /** 分析
         在函数 getFirstMoreThan20 中,通过第一步发现v1(a*5)大于20.直接返回
         此时返现第二个参数 b*5成了无效操作,因此试想内否让b*5延迟执行
         即只有当a*5不满足条件时才会发生b*5操作,这无疑会提升时间性能
         */
        
        // 改造函数1
        func getFirstMoreThan20_1(_ v1: Int, _ v2: ()->Int)->Int {
            return v1 > 20 ? v1 : v2()
        }
        
        // 此时的调用变成了这样,能满足我们说的v2延迟执行,即v1满足了条件,v2的b*5不会发生
        // 不过该做法看起来不够优雅,还必须要求我们写{},swift提供的自动闭包技术可以优化该操作
        getFirstMoreThan20_1(a * 5, {b * 5})
        getFirstMoreThan20_1(a * 5) {b * 5}
        
        
        // 改造函数2(在v2闭包之前添加@autoclosure关键字)
        func getFirstMoreThan20_2(_ v1: Int, _ v2: @autoclosure ()->Int)->Int {
            return v1 > 20 ? v1 : v2()
        }
        
        // 此时调用变得和普通函数没有区别,也看不出 b * 5 会延迟执行
        // 但实际上是隐形优化,对外暴露的接口很友好,内部确做了优化操作
        // 该操作相当于@autoclosure会在编译阶段自动包装代码,过程如下
        // b * 5 --> {b * 5} --> {()->Int in b * 5}
        // 这就是自动闭包的优雅之处
        getFirstMoreThan20_2(a * 5, b * 5)
    }
    test_auto_closure()
上一篇下一篇

猜你喜欢

热点阅读