最近在學(xué)習(xí)RxSwift>>>,大量接觸閉包的使用吠谢,趁著下班前土童,翻譯了Swift官方文檔中閉包那經(jīng)常使用的一部分,當(dāng)作復(fù)習(xí)工坊。
閉包表達(dá)式語法
Closure表達(dá)式語法具有以下一般形式:
{ (parameters) -> return type in
statements
}
閉包表達(dá)式語法中的參數(shù)可以是輸入?yún)?shù)献汗,但它們不能有默認(rèn)值。如果命名可變參數(shù)王污,可以使用變量參數(shù)罢吃。元組也可以用作參數(shù)類型和返回類型。
看一個(gè)例子:將一個(gè)數(shù)組排序昭齐。之前的做法是:定義一個(gè)方法尿招,傳入數(shù)組,返回排序結(jié)果阱驾。
let names = ["Chris", "Alex", "Ewa", "Barry", "Daniella"]
func backward(_ s1: String, _ s2: String) -> Bool {
return s1 > s2
}
var reversedNames = names.sorted(by: backward)
使用閉包以后:
reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in
return s1 > s2
})
由于閉包的體積非常短就谜,所以它甚至可以寫在一行上:
reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in return s1 > s2 } )
從上下文推斷類型
因?yàn)榕判蜷]包作為參數(shù)傳遞給方法,所以Swift可以推斷它的參數(shù)類型和它返回的值的類型里覆。排序后的(by :)方法在字符串?dāng)?shù)組上調(diào)用丧荐,所以它的參數(shù)必須是類型(String,String) - > Bool的函數(shù)喧枷。這意味著(String虹统,String)和Bool類型不需要作為閉包表達(dá)式定義的一部分寫入。由于可以推斷所有類型隧甚,因此也可以省略返回箭頭( - >)和參數(shù)名稱周圍的括號(hào):
reversedNames = names.sorted(by: { s1, s2 in return s1 > s2 } )
在將閉包作為內(nèi)聯(lián)閉包表達(dá)式傳遞給函數(shù)或方法時(shí)车荔,始終可以推斷參數(shù)類型和返回類型。因此呻逆,當(dāng)閉包用作函數(shù)或方法參數(shù)時(shí)夸赫,您絕不需要以最充分的形式編寫內(nèi)聯(lián)閉包。
來自單表達(dá)式閉包的隱式返回
通過在聲明中省略return關(guān)鍵字咖城,單表達(dá)式閉包可隱式返回其單個(gè)表達(dá)式的結(jié)果,如前面示例的此版本所示:
reversedNames = names.sorted(by: { s1, s2 in s1 > s2 } )
這里呼奢,排序后的(by :)方法參數(shù)的函數(shù)類型清楚地表明了閉包必須返回一個(gè)Bool值宜雀。由于閉包的主體包含一個(gè)返回Bool值的單個(gè)表達(dá)式(s1> s2),因此不存在歧義握础,并且可以省略return關(guān)鍵字辐董。
速記參數(shù)名稱
Swift自動(dòng)為內(nèi)聯(lián)閉包提供速記參數(shù)名稱,這些名稱可用于通過名稱$ 0禀综,$ 1简烘,$ 2等引用閉包參數(shù)的值苔严。
reversedNames = names.sorted(by: { $0 > $1 } )
尾隨閉包
如果需要將閉包表達(dá)式作為函數(shù)的最終參數(shù)傳遞給函數(shù),并且閉包表達(dá)式很長孤澎,那么將其作為尾部閉包編寫可能會(huì)很有用届氢。尾隨閉包在函數(shù)調(diào)用的括號(hào)后面寫入,盡管它仍然是該函數(shù)的參數(shù)覆旭。在使用尾隨閉包語法時(shí)退子,不要將閉包的參數(shù)標(biāo)簽作為函數(shù)調(diào)用的一部分寫入。
func someFunctionThatTakesAClosure(closure: () -> Void) {
// function body goes here
}
// Here's how you call this function without using a trailing closure:
someFunctionThatTakesAClosure(closure: {
// closure's body goes here
})
// Here's how you call this function with a trailing closure instead:
someFunctionThatTakesAClosure() {
// trailing closure's body goes here
}
上面的Closure Expression Syntax部分中的字符串排序閉包可以在排序的(by :)方法的括號(hào)之外寫為尾部閉包:
reversedNames = names.sorted() { $0 > $1 }
如果提供閉包表達(dá)式作為函數(shù)或方法的唯一參數(shù)型将,并且將該表達(dá)式作為尾隨閉包提供寂祥,則在調(diào)用該函數(shù)時(shí),無需在函數(shù)或方法名稱后面編寫一對(duì)括號(hào)():
reversedNames = names.sorted { $0 > $1 }
當(dāng)封閉足夠長以至于不可能將它寫在一行上時(shí)七兜,尾隨封閉非常有用丸凭。作為一個(gè)例子,Swift的數(shù)組類型有一個(gè)map(_ :)方法腕铸,它將一個(gè)閉包表達(dá)式作為其單個(gè)參數(shù)贮乳。對(duì)于數(shù)組中的每個(gè)項(xiàng)目都調(diào)用一次閉包,并為該項(xiàng)目返回一個(gè)替代映射值(可能是某種其他類型)恬惯。映射的性質(zhì)和返回值的類型留給閉包來指定向拆。