前言
閉包是功能性自包含模塊,可以在代碼中被傳遞和使用。 Swift 中的閉包與 C 和 Objective-C中的 blocks 以及其他一些編程語言中的 lambdas 比較相似。下面這篇文章就來詳細介紹了關于Swift 3.0中的閉包,感興趣的一起來看看吧。
開始
閉包的書寫格式如下:
1
2
3
|
{ (parameters) -> return type in statements } |
如
1
|
reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in return s1 > s2 } ) |
使用的時候可以簡化為
1
|
reversedNames = names.sorted(by: { s1, s2 in return s1 > s2 } ) |
也可以簡化為
1
|
reversedNames = names.sorted(by: { s1, s2 in s1 > s2 } ) |
甚至可以簡化為
1
|
reversedNames = names.sorted(by: { $0 > $1 } ) |
$0為第一個參數,以此類推
返回布爾值的話可以直接給一個判斷符號,如
1
|
reversedNames = names.sorted(by: >) |
尾部閉包
1
|
reversedNames = names.sorted() { $0 > $1 } |
或者(無其他參數的情況下)
1
|
reversedNames = names.sorted { $0 > $1 } |
這兩個方式均可
捕捉值
如下代碼,閉包可以獲取和修改其周圍的變量
1
2
3
4
5
6
7
8
|
func makeIncrementer(forIncrement amount: Int) -> () -> Int { var runningTotal = 0 func incrementer() -> Int { runningTotal += amount return runningTotal } return incrementer } |
以上函數返回一個閉包,閉包里添加了讀取和修改閉包的外部變量runningTotal
1
2
3
4
5
6
7
|
let incrementByTen = makeIncrementer(forIncrement: 10) incrementByTen() // returns a value of 10 incrementByTen() // returns a value of 20 incrementByTen() // returns a value of 30 |
閉包引用類型
可以通過用這種方式引用閉包并且調用:
1
2
|
let alsoIncrementByTen = incrementByTen alsoIncrementByTen() |
@escaping
傳遞給函數的閉包如果不是在函數內調用,而是在函數內用外部變量保存當前的閉包,在合適的時間再進行調用,這是就需要在閉包參數前加入@escaping關鍵字,不然編譯器會報錯。
比較好理解的就是經常用到的網絡請求,請求完成才執行完成的閉包。
官方的例子如下:
1
2
3
4
5
6
7
8
9
|
//需要加@escaping的情況 var completionHandlers: [() -> Void] = [] func someFunctionWithEscapingClosure(completionHandler: @escaping () -> Void) { completionHandlers.append(completionHandler) } //直接在函數內調用傳入的閉包則不需要@escaping func someFunctionWithNonescapingClosure(closure: () -> Void) { closure() } |
@autoclosure 自動閉包
1
2
3
4
5
6
|
// customersInLine is ["Alex", "Ewa", "Barry", "Daniella"] func serve(customer customerProvider: () -> String) { print( "Now serving \(customerProvider())!" ) } serve(customer: { customersInLine. remove (at: 0) } ) // Prints "Now serving Alex!" |
如上代碼所示,我們加入一個返回類型為String的閉包,需要在外面加上{},為了書寫方便, 加上@autoclosure關鍵字,那么這個對{}就可以省略了。
1
2
3
4
5
6
|
// customersInLine is ["Ewa", "Barry", "Daniella"] func serve(customer customerProvider: @autoclosure () -> String) { print( "Now serving \(customerProvider())!" ) } serve(customer: customersInLine. remove (at: 0)) // Prints "Now serving Ewa!" |
編譯器會幫我們標記這行代碼為閉包,這段代碼不會馬上被調用,而是當成閉包在函數里調用的時候才被調用。
參考英語原文:
https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/Closures.html#//apple_ref/doc/uid/TP40014097-CH11-ID94
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者使用Swift能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。
原文鏈接:http://devlong.com/2016/11/06/closures/