swift 闭包本质,闭包表达式,尾随闭包

1. 闭包

  • 一个函数和它所捕获的变量/常量环境组合起来,称为闭包

    • 一般指定义在函数内部的函数
    • 一般它所捕获的是外层函数的局部变量/常量
    typealias fn = (Int) -> Int
    func getFn() -> fn{
    	var count = 0
    		func sum(_ i: Int) -> Int{
        		count += i
        		return count
    		}
    	return sum
    }
    
    var f1 = getFn()
    
    f1(1)
    f1(1)
    f1(1)
    f1(1)
    

    结果:
    image
    解释:
    闭包能够使用其外层函数的局部变量,所以函数值能够增加
    本质:
    编译器给sum函数外层getFn函数的count属性分配了堆空间,所以count变量不会在getFn函数执行完后销毁,因此sum函数能够对其进行访问
    分配内存结构: 类似于给class分配的堆空间结构
    image

    • 可以把闭包想象成一个对象的实例

      • 内存在堆空间
      • 捕获的局部变量/常量就是对象的成员
      • 组成闭包的函数就是类内部定义的方法
        类似与class的形式:
      class Closure{
      var count = 0
      	func sum(_ i:Int) -> Int{
      		count += i
      		return count
      	}
      }
      
      var c1 = Closure()
      c1.sum(1)
      c1.sum(1)
      c1.sum(1)
      

2. 闭包表达式

  • 语法:
    {
    	(参数列表) -> 返回值类型 in
    	函数体代码
    }
    
  • 简写:
    func exec(v1:Int, v2:Int, fn:(Int, Int) -> Int){
    	print(fn(v1, v2))
    }
    // 完整写法:
    exec(v1:10, v2:20, fn:{
    	(a1:Int, a2:Int) -> Int in
    	return a1 + a2
    })
    // 简写1
    exec(v1:10, v2:20, fn:{
    	a1, a2 in return a1 + a2
    })
    // 简写2
    exec(v1:10, v2:20,fn:{
    	a1,a2 in a1 + a2
    })
    // 简写3
    exec(v1:10, v2:20, fn:{ $0 + $1 })
    // 简写4
    exec(v1:10, v2:20, fn: + )
    // 尾随闭包: 是一个被书写在函数调用括号外面(后面)的闭包表达式
    // 如果一个很长的闭包表达式作为一个函数的 最后一个 实参,使用尾随闭包可以增强程序的可读性
    exec(v1:10, v2:20){ $0 + $1 }
    // 如果函数只有一个参数,且类型是函数类型,可以省略括号
    func add(fn: (Int,Int) -> Int){
    	print(fn(1, 2))
    }
    add{ $0 + $1 }
    // 省略参数
    add{ _,_ in 10 }  //省略掉参数,固定返回10
    
posted @ 2022-03-16 09:57  基地您  阅读(350)  评论(0编辑  收藏  举报