swift中defer幾個(gè)簡單的使用場景詳解
前言
最近準(zhǔn)備把 swift 文檔再掃一遍,發(fā)現(xiàn)了 defer 這個(gè)關(guān)鍵字,defer 是個(gè)非常重要的 swift 語言特征,恕本人愚鈍,以前還從來沒有用過這個(gè)呢~ 簡單地列一下這個(gè)東西有哪些可以用得上的情景吧~~話不多說了,來一起看看詳細(xì)的介紹吧。
defer 是干什么用的
很簡單,用一句話概括,就是 defer block 里的代碼會(huì)在函數(shù) return 之前執(zhí)行,無論函數(shù)是從哪個(gè)分支 return 的,還是有 throw,還是自然而然走到最后一行。
這個(gè)關(guān)鍵字就跟 Java 里的 try-catch-finally 的 finally 一樣,不管 try catch 走哪個(gè)分支,它都會(huì)在函數(shù) return 之前執(zhí)行。而且它比 Java 的 finally 還更強(qiáng)大的一點(diǎn)是,它可以獨(dú)立于 try catch 存在,所以它也可以成為整理函數(shù)流程的一個(gè)小幫手。在函數(shù) return 之前無論如何都要做的處理,可以放進(jìn)這個(gè) block 里,讓代碼看起來更干凈一些~
下面是 swift 文檔上的例子:
var fridgeIsOpen = false let fridgeContent = ["milk", "eggs", "leftovers"] func fridgeContains(_ food: String) -> Bool { fridgeIsOpen = true defer { fridgeIsOpen = false } let result = fridgeContent.contains(food) return result } fridgeContains("banana") print(fridgeIsOpen)
這個(gè)例子里執(zhí)行的順序是,先 fridgeIsOpen = true ,然后是函數(shù)體正常的流程,最后在 return 之前執(zhí)行 fridgeIsOpen = false 。
幾個(gè)簡單的使用場景
try catch 結(jié)構(gòu)
最典型的場景,我想也是 defer 這個(gè)關(guān)鍵字誕生的主要原因吧:
func foo() { defer { print("finally") } do { throw NSError() print("impossible") } catch { print("handle error") } }
不管 do block 是否 throw error,有沒有 catch 到,還是 throw 出去了,都會(huì)保證在整個(gè)函數(shù) return 前執(zhí)行 defer 。在這個(gè)例子里,就是先 print 出 "handle error" 再 print 出 "finally"。
do block 里也可以寫 defer :
do { defer { print("finally") } throw NSError() print("impossible") } catch { print("handle error") }
那么它執(zhí)行的順序就會(huì)是在 catch block 之前,也就是先 print 出 "finally" 再 print 出 "handle error"。
清理工作、回收資源
跟 swift 文檔舉的例子類似, defer 一個(gè)很適合的使用場景就是用來做清理工作。文件操作就是一個(gè)很好的例子:
關(guān)閉文件
func foo() { let fileDescriptor = open(url.path, O_EVTONLY) defer { close(fileDescriptor) } // use fileDescriptor... }
這樣就不怕哪個(gè)分支忘了寫,或者中間 throw 個(gè) error,導(dǎo)致 fileDescriptor 沒法正常關(guān)閉。還有一些類似的場景:
dealloc 手動(dòng)分配的空間
func foo() { let valuePointer = UnsafeMutablePointer<T>.allocate(capacity: 1) defer { valuePointer.deallocate(capacity: 1) } // use pointer... }
加/解鎖:下面是 swift 里類似 Objective-C 的 synchronized block 的一種寫法,可以使用任何一個(gè) NSObject 作 lock
func foo() { objc_sync_enter(lock) defer { objc_sync_exit(lock) } // do something... }
像這種成對調(diào)用的方法,可以用 defer 把它們放在一起,一目了然。
調(diào) completion block
這是一個(gè)讓我感覺“如果當(dāng)時(shí)知道 defer ”就好了的場景,就是有時(shí)候一個(gè)函數(shù)分支比較多,可能某個(gè)小分支 return 之前就忘了調(diào) completion block,結(jié)果藏下一個(gè)不易發(fā)現(xiàn)的 bug。用 defer 就可以不用擔(dān)心這個(gè)問題了:
func foo(completion: () -> Void) { defer { self.isLoading = false completion() } guard error == nil else { return } // handle success }
有時(shí)候 completion 要根據(jù)情況傳不同的參數(shù),這時(shí) defer 就不好使了。不過如果 completion block 被存下來了,我們還是可以用它來確保執(zhí)行后能釋放:
func foo() { defer { self.completion = nil } if (succeed) { self.completion(.success(result)) } else { self.completion(.error(error)) } }
調(diào) super 方法
有時(shí)候 override 一個(gè)方法,主要目的是在 super 方法之前做一些準(zhǔn)備工作,比如 UICollectionViewLayout 的 prepare(forCollectionViewUpdates:) ,那么我們就可以把調(diào)用 super 的部分放在 defer 里:
func override foo() { defer { super.foo() } // some preparation before super.foo()... }
一些細(xì)節(jié)
任意 scope 都可以有 defer
雖然大部分的使用場景是在函數(shù)里,不過理論上任何一個(gè) { } 之間都是可以寫 defer 的。比如一個(gè)普通的循環(huán):
var sumOfOdd = 0 for i in 0...10 { defer { print("Look! It's \(i)") } if i % 2 == 0 { continue } sumOfOdd += i }
continue 或者 break 都不會(huì)妨礙 defer 的執(zhí)行。甚至一個(gè)平白無故的 closure 里也可以寫 defer :
{ defer { print("bye!") } print("hello!") }
就是這樣沒什么意義就是了……
必須執(zhí)行到 defer 才會(huì)觸發(fā)
假設(shè)有這樣一個(gè)問題:一個(gè) scope 里的 defer 能保證一定會(huì)執(zhí)行嗎? 答案是否……比如下面這個(gè)例子:
func foo() throws { do { throw NSError() print("impossible") } defer { print("finally") } } try?foo()
不會(huì)執(zhí)行 defer,不會(huì) print 任何東西。這個(gè)故事告訴我們,至少要執(zhí)行到 defer 這一行,它才保證后面會(huì)觸發(fā)。同樣道理,提前 return 也是一樣不行的:
func foo() { guard false else { return } defer { print("finally") } }
多個(gè) defer
一個(gè) scope 可以有多個(gè) defer,順序是像棧一樣倒著執(zhí)行的:每遇到一個(gè) defer 就像壓進(jìn)一個(gè)棧里,到 scope 結(jié)束的時(shí)候,后進(jìn)棧的先執(zhí)行。如下面的代碼,會(huì)按 1、2、3、4、5、6 的順序 print 出來。
func foo() { print("1") defer { print("6") } print("2") defer { print("5") } print("3") defer { print("4") } }
但是我強(qiáng)烈建議不要這么寫。我是建議一個(gè) scope 里不要有多個(gè) defer,感覺除了讓讀代碼的人感覺混亂之外沒有什么好處。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對我們的支持。
上一篇:Swift利用Decodable解析JSON的一個(gè)小問題詳解
欄 目:Swift
下一篇:沒有了
本文標(biāo)題:swift中defer幾個(gè)簡單的使用場景詳解
本文地址:http://mengdiqiu.com.cn/a1/Swift/11967.html
您可能感興趣的文章
- 01-11Swift利用Decodable解析JSON的一個(gè)小問題詳解
- 01-11Swift中defer關(guān)鍵字推遲執(zhí)行示例詳解
- 01-11Swift中初始化init的方法小結(jié)
- 01-11Swift中定義單例的方法實(shí)例
- 01-11Swift利用純代碼實(shí)現(xiàn)時(shí)鐘效果實(shí)例代碼
- 01-11Swift中排序算法的簡單取舍詳解
- 01-11Swift如何為設(shè)置中心添加常用功能
- 01-11Swift Json實(shí)例詳細(xì)解析
- 01-11Swift利用指紋識(shí)別或面部識(shí)別為應(yīng)用添加私密保護(hù)功能
- 01-11Swift 4.0中如何引用3.0的第三方庫


閱讀排行
本欄相關(guān)
- 01-11Swift利用Decodable解析JSON的一個(gè)小問題
- 01-11swift中defer幾個(gè)簡單的使用場景詳解
- 01-11Swift中初始化init的方法小結(jié)
- 01-11Swift中defer關(guān)鍵字推遲執(zhí)行示例詳解
- 01-11Swift利用純代碼實(shí)現(xiàn)時(shí)鐘效果實(shí)例代碼
- 01-11Swift中定義單例的方法實(shí)例
- 01-11Swift中排序算法的簡單取舍詳解
- 01-11Swift Json實(shí)例詳細(xì)解析
- 01-11Swift如何為設(shè)置中心添加常用功能
- 01-11Swift利用指紋識(shí)別或面部識(shí)別為應(yīng)用添
隨機(jī)閱讀
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子