swift閉包的@noescape與@autoclosure

對于程序員來說,如果你精通了閉包技術(shù)晦炊,那么毫無疑問璧瞬,你的技術(shù)水平會得到大踏步的前進(jìn)。
我們這次的主題是聊聊swift閉包的幾個概念Capturing Values麻裁、Nonescaping Closures和``Autoclosures`箍镜。

Capturing Values

在電影超人3當(dāng)中,Gus Gorman修改了公司的程序煎源,在給每人結(jié)算工資的時候色迂,偷偷的將其中微小的一部分轉(zhuǎn)到自己的賬戶下,從而累積了巨額財富手销。我們今天就用閉包的Capturing Values性質(zhì)來實(shí)現(xiàn)這個算法燎潮。值得一提的是梧却,原作者在算法中考慮了加班工資的部分,因考慮具體國情,根據(jù)實(shí)際情況泛豪,我已經(jīng)把加班工資的相關(guān)代碼去掉了。

struct PennyShaver {
    static var stolenSoFar = 0.0
    static var numberOfPeopleStolenFrom = 0
    var amountToSteal = 0.01
    
    func createPaymentCalculator() -> (Double,Double) -> Double {
        func calculatePayroll(hourlyRate: Double,hours: Double) -> Double {
            let payrollAmount = hourlyRate * hours
            
            PennyShaver.stolenSoFar += amountToSteal
            PennyShaver.numberOfPeopleStolenFrom += 1
            
            return payrollAmount - amountToSteal
        }
        return calculatePayroll
    }
}

代碼超級簡單莱坎,我們在struct中定義了createPaymentCalculator方法玫氢,該方法會返回一個閉包工資計算器calculatePayroll,該計算器用公式: 工作時間 * 每小時費(fèi)用 來計算每個人的工資讲仰。

這個計算器有一個厲害的特性慕趴,就是它能夠捕捉context(PennyShaver)中的變量stolenSoFar、numberOfPeopleStolenFrom鄙陡、amountToSteal用來計算 偷了多少人的錢 以及 偷到多少錢冕房。于是我們可以這樣展開“偷錢”工作。

var pennyShaver = PennyShaver(amountToSteal: 0.10)
let calculator = pennyShaver.createPaymentCalculator()
let amount = calculator(70,40)    // 2799.9 偷到了一分錢

我們可以看到趁矾,閉包可以把變量amountToSteal帶出該變量所屬的正常生命周期之外耙册,仍然進(jìn)行讀寫維護(hù)操作,這個特性就是Capturing Values.這是閉包最重要的特性毫捣,也是閉包之所以稱之為閉包的重要原因详拙。

Nonescaping Closures

在聊Nonescaping Closures(非逃逸型閉包)之前,咱們都先說說escape型閉包蔓同。如果一個閉包被作為參數(shù)傳遞給一個函數(shù)饶辙,如果該函數(shù)允許這個閉包在函數(shù)自己的生命周期結(jié)束以后仍然可以被使用,那么我們就說這個作為參數(shù)的閉包是:“可逃逸的”斑粱。舉一個例子就更清楚了弃揽。

var callbackListenerList: [(Bool,String) -> Void] = []

func registerListenerCallbacks(callback: (Bool,String) -> Void) {
    callbackListenerList.append(callback)
}

registerListenerCallbacks({(_,message) -> Void in
    print(message)
})

registerListenerCallbacks函數(shù)接收一個閉包,接著它將這個閉包append到一個數(shù)組中,所以即使這個函數(shù)的生命周期結(jié)束了矿微。我們在未來的某一時刻讓然可以利用這個閉包來工作痕慢。比方說像下面這樣。

let callback = callbackListenerList.first!
callback(true,"Hello,world")

這個時候涌矢,我們稱參數(shù)閉包“逃逸了”掖举,逃逸型閉包非常適用于異步場景。

如果你此時屏蔽這種逃逸現(xiàn)象(禁止閉包在回調(diào)數(shù)組中工作)娜庇,那么就可以使用關(guān)鍵字@noescape來修飾閉包參數(shù)拇泛,這個關(guān)鍵字就是像編譯器聲明,函數(shù)結(jié)束以后思灌,我不在需要這個閉包了俺叭。

func registerListenerCallbacks(@noescape callback: (Bool,String) -> Void) {
    callbackListenerList.append(callback) // error !L┏ァOㄊ亍!:孽恕裕照!
}

如果我們這樣修改了registerListenerCallbacks方法,那么就會收到編譯器的錯誤提示调塌。通過@noescape`,編譯器就會非常明確自己應(yīng)該如何進(jìn)一步優(yōu)化代碼了晋南,否則編譯器在沒有收到我們明確的態(tài)度之前,它必須給我們保留各種可能的余地羔砾。

Autoclosures

在我們傳遞閉包的時候负间,語法要求我們給閉包的外側(cè)添加一對{},就像下面這個代碼片段所描述的那樣姜凄。

func printString(f: () -> ()) {
    f()
}

printString({print("dddd")})

函數(shù)printString接收一個閉包作為參數(shù)政溃,當(dāng)我們調(diào)用printString的時候,我們需要用到一個對括號{print("dddd")}态秧,如果我們使用@autoclosure來修飾參數(shù)的話董虱,我們可以省略這對括號。

func printString(@autoclosure f: () -> ()) {
    f()
}
printString(print("dddd"))

@autoclosure可以有效的簡化我們的代碼申鱼,但隨之而來的問題是愤诱,它降低了代碼的可讀性,所以我們要在簡潔與可讀性之間選擇好一個平衡點(diǎn)捐友。

另外值得注意點(diǎn)的是@autoclosure默認(rèn)是nonescape的淫半,如果我們需要在@autoclosure的基礎(chǔ)上做進(jìn)一步的說明,指明這個閉包參數(shù)是可逃逸的楚殿。

func printString(@autoclosure(escaping) f: () -> ()) {
    f()
}

總結(jié)

雖然有些內(nèi)容我們在實(shí)際的工作中并不經(jīng)常使用撮慨,但值得我們注意的一點(diǎn)是:有些內(nèi)容是在潛移默化的影響著我們,例如:審美脆粥,我們也很難說清楚我們自己的審美標(biāo)準(zhǔn)是從何而來砌溺,技術(shù)也同樣,想更上一層樓变隔,我們就需要積累那些可能會潛移默化影響我們的東西规伐。

內(nèi)容參考:

Swift Closures?—?Everyday Gems Part 1 of 2
Swift Closures?—?Everyday Gems Part 2 of 2

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市匣缘,隨后出現(xiàn)的幾起案子猖闪,更是在濱河造成了極大的恐慌,老刑警劉巖肌厨,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件培慌,死亡現(xiàn)場離奇詭異,居然都是意外死亡柑爸,警方通過查閱死者的電腦和手機(jī)吵护,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來表鳍,“玉大人馅而,你說我怎么就攤上這事∑┦ィ” “怎么了瓮恭?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵,是天一觀的道長厘熟。 經(jīng)常有香客問我屯蹦,道長,這世上最難降的妖魔是什么绳姨? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任颇玷,我火速辦了婚禮,結(jié)果婚禮上就缆,老公的妹妹穿的比我還像新娘帖渠。我一直安慰自己,他們只是感情好竭宰,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布空郊。 她就那樣靜靜地躺著,像睡著了一般切揭。 火紅的嫁衣襯著肌膚如雪狞甚。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天廓旬,我揣著相機(jī)與錄音哼审,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛涩盾,可吹牛的內(nèi)容都是我干的十气。 我是一名探鬼主播,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼春霍,長吁一口氣:“原來是場噩夢啊……” “哼砸西!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起址儒,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤芹枷,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后莲趣,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體鸳慈,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年喧伞,在試婚紗的時候發(fā)現(xiàn)自己被綠了蝶涩。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡絮识,死狀恐怖绿聘,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情次舌,我是刑警寧澤熄攘,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站彼念,受9級特大地震影響挪圾,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜逐沙,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一哲思、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧吩案,春花似錦棚赔、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至残揉,卻和暖如春胧后,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背抱环。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工壳快, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留纸巷,地道東北人。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓眶痰,卻偏偏與公主長得像瘤旨,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子凛驮,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容