Swift 3.0基础学习之闭包

2020-01-08 23:35:07王振洲

前言

闭包是功能性自包含模块,可以在代码中被传递和使用。 Swift 中的闭包与 C 和 Objective-C中的 blocks 以及其他一些编程语言中的 lambdas 比较相似。下面这篇文章就来详细介绍了关于Swift 3.0中的闭包,感兴趣的一起来看看吧。

开始

闭包的书写格式如下:


{ (parameters) -> return type in
statements
}


reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in return s1 > s2 } )

使用的时候可以简化为


reversedNames = names.sorted(by: { s1, s2 in return s1 > s2 } )

也可以简化为


reversedNames = names.sorted(by: { s1, s2 in s1 > s2 } )

甚至可以简化为


reversedNames = names.sorted(by: { $0 > $1 } )

$0为第一个参数,以此类推

返回布尔值的话可以直接给一个判断符号,如


reversedNames = names.sorted(by: >)

尾部闭包


reversedNames = names.sorted() { $0 > $1 }

或者(无其他参数的情况下)


reversedNames = names.sorted { $0 > $1 }

这两个方式均可

捕捉值

如下代码,闭包可以获取和修改其周围的变量


func makeIncrementer(forIncrement amount: Int) -> () -> Int {
 var runningTotal = 0
 func incrementer() -> Int {
 runningTotal += amount
 return runningTotal
 }
 return incrementer
}

以上函数返回一个闭包,闭包里添加了读取和修改闭包的外部变量runningTotal


let incrementByTen = makeIncrementer(forIncrement: 10)
incrementByTen()
// returns a value of 10
incrementByTen()
// returns a value of 20
incrementByTen()
// returns a value of 30

闭包引用类型

可以通过用这种方式引用闭包并且调用:


let alsoIncrementByTen = incrementByTen
alsoIncrementByTen()

@escaping

传递给函数的闭包如果不是在函数内调用,而是在函数内用外部变量保存当前的闭包,在合适的时间再进行调用,这是就需要在闭包参数前加入@escaping关键字,不然编译器会报错。

比较好理解的就是经常用到的网络请求,请求完成才执行完成的闭包。

官方的例子如下:


//需要加@escaping的情况
var completionHandlers: [() -> Void] = []
func someFunctionWithEscapingClosure(completionHandler: @escaping () -> Void) {
 completionHandlers.append(completionHandler)
}
//直接在函数内调用传入的闭包则不需要@escaping
func someFunctionWithNonescapingClosure(closure: () -> Void) {
 closure()
}