欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

swift中defer幾個簡單的使用場景詳解

 更新時間:2018年03月05日 08:36:26   投稿:daisy  
在Swift 2.0中,Apple提供了defer關鍵字,讓我們可以實現(xiàn)同樣的效果,這篇文章主要介紹了關于swift中defer幾個簡單的使用場景的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用defer具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習吧。

前言

最近準備把 swift 文檔再掃一遍,發(fā)現(xiàn)了 defer 這個關鍵字,defer 是個非常重要的 swift 語言特征,恕本人愚鈍,以前還從來沒有用過這個呢~ 簡單地列一下這個東西有哪些可以用得上的情景吧~~話不多說了,來一起看看詳細的介紹吧。

defer 是干什么用的

很簡單,用一句話概括,就是 defer block 里的代碼會在函數(shù) return 之前執(zhí)行,無論函數(shù)是從哪個分支 return 的,還是有 throw,還是自然而然走到最后一行。

這個關鍵字就跟 Java 里的 try-catch-finally 的 finally 一樣,不管 try catch 走哪個分支,它都會在函數(shù) return 之前執(zhí)行。而且它比 Java 的 finally 還更強大的一點是,它可以獨立于 try catch 存在,所以它也可以成為整理函數(shù)流程的一個小幫手。在函數(shù) return 之前無論如何都要做的處理,可以放進這個 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)

這個例子里執(zhí)行的順序是,先 fridgeIsOpen = true ,然后是函數(shù)體正常的流程,最后在 return 之前執(zhí)行 fridgeIsOpen = false 。

幾個簡單的使用場景

try catch 結(jié)構(gòu)

最典型的場景,我想也是 defer 這個關鍵字誕生的主要原因吧:

func foo() {
 defer {
 print("finally")
 }
 do {
 throw NSError()
 print("impossible")
 } catch {
 print("handle error")
 }
}

不管 do block 是否 throw error,有沒有 catch 到,還是 throw 出去了,都會保證在整個函數(shù) return 前執(zhí)行 defer 。在這個例子里,就是先 print 出 "handle error" 再 print 出 "finally"。

do block 里也可以寫 defer :

do {
 defer {
 print("finally")
 }
 throw NSError()
 print("impossible")
} catch {
 print("handle error")
}

那么它執(zhí)行的順序就會是在 catch block 之前,也就是先 print 出 "finally" 再 print 出 "handle error"。

清理工作、回收資源

跟 swift 文檔舉的例子類似, defer 一個很適合的使用場景就是用來做清理工作。文件操作就是一個很好的例子:

關閉文件

func foo() {
 let fileDescriptor = open(url.path, O_EVTONLY)
 defer {
 close(fileDescriptor)
 }
 // use fileDescriptor...
}

這樣就不怕哪個分支忘了寫,或者中間 throw 個 error,導致 fileDescriptor 沒法正常關閉。還有一些類似的場景:

dealloc 手動分配的空間

func foo() {
 let valuePointer = UnsafeMutablePointer<T>.allocate(capacity: 1)
 defer {
 valuePointer.deallocate(capacity: 1)
 }
 // use pointer...
}

加/解鎖:下面是 swift 里類似 Objective-C 的 synchronized block 的一種寫法,可以使用任何一個 NSObject 作 lock

func foo() {
 objc_sync_enter(lock)
 defer { 
 objc_sync_exit(lock)
 }
 // do something...
}

像這種成對調(diào)用的方法,可以用 defer 把它們放在一起,一目了然。

調(diào) completion block

這是一個讓我感覺“如果當時知道 defer ”就好了的場景,就是有時候一個函數(shù)分支比較多,可能某個小分支 return 之前就忘了調(diào) completion block,結(jié)果藏下一個不易發(fā)現(xiàn)的 bug。用 defer 就可以不用擔心這個問題了:

func foo(completion: () -> Void) {
 defer {
 self.isLoading = false
 completion()
 }
 guard error == nil else { return } 
 // handle success
}

有時候 completion 要根據(jù)情況傳不同的參數(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 方法

有時候 override 一個方法,主要目的是在 super 方法之前做一些準備工作,比如 UICollectionViewLayout 的 prepare(forCollectionViewUpdates:) ,那么我們就可以把調(diào)用 super 的部分放在 defer 里:

func override foo() {
 defer {
 super.foo()
 }
 // some preparation before super.foo()...
}

一些細節(jié)

任意 scope 都可以有 defer

雖然大部分的使用場景是在函數(shù)里,不過理論上任何一個 { } 之間都是可以寫 defer 的。比如一個普通的循環(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 都不會妨礙 defer 的執(zhí)行。甚至一個平白無故的 closure 里也可以寫 defer :

{
 defer { print("bye!") }
 print("hello!")
}

就是這樣沒什么意義就是了……

必須執(zhí)行到 defer 才會觸發(fā)
假設有這樣一個問題:一個 scope 里的 defer 能保證一定會執(zhí)行嗎? 答案是否……比如下面這個例子:

func foo() throws {
 do {
 throw NSError()
 print("impossible")
 }
 defer {
 print("finally")
 }
}
try?foo()

不會執(zhí)行 defer,不會 print 任何東西。這個故事告訴我們,至少要執(zhí)行到 defer 這一行,它才保證后面會觸發(fā)。同樣道理,提前 return 也是一樣不行的:

func foo() {
 guard false else { return }
 defer {
 print("finally")
 }
}

多個 defer

一個 scope 可以有多個 defer,順序是像棧一樣倒著執(zhí)行的:每遇到一個 defer 就像壓進一個棧里,到 scope 結(jié)束的時候,后進棧的先執(zhí)行。如下面的代碼,會按 1、2、3、4、5、6 的順序 print 出來。

func foo() {
 print("1")
 defer {
 print("6")
 }
 print("2")
 defer {
 print("5")
 }
 print("3")
 defer {
 print("4")
 }
}

但是我強烈建議不要這么寫。我是建議一個 scope 里不要有多個 defer,感覺除了讓讀代碼的人感覺混亂之外沒有什么好處。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

相關文章

  • Swift教程之閉包詳解

    Swift教程之閉包詳解

    這篇文章主要介紹了Swift教程之閉包詳解,閉包可以在上下文的范圍內(nèi)捕獲、存儲任何被定義的常量和變量引用,因這些常量和變量的封閉性,而命名為“閉包(Closures)”,需要的朋友可以參考下
    2015-01-01
  • 淺談swift 4.0中private所發(fā)生的變化

    淺談swift 4.0中private所發(fā)生的變化

    Swift 4是蘋果計劃于2017年秋季推出的最新版本,其主要重點是提供與Swift 3代碼的源兼容性,并努力實現(xiàn)ABI穩(wěn)定性。下面這篇文章主要給大家介紹了關于swift 4.0中private所發(fā)生的一些變化,需要的朋友可以參考下。
    2017-12-12
  • Swift4使用GCD實現(xiàn)計時器

    Swift4使用GCD實現(xiàn)計時器

    這篇文章主要為大家詳細介紹了Swift4使用GCD實現(xiàn)計時器,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • Swift下使用UICollectionView 實現(xiàn)長按拖拽功能

    Swift下使用UICollectionView 實現(xiàn)長按拖拽功能

    拖拽排序是新聞類的App可以說是必有的交互設計,如今日頭條,網(wǎng)易新聞等。這篇文章主要介紹了Swift下使用UICollectionView 長按拖拽功能,需要的朋友可以參考下
    2017-03-03
  • swift 字符串String的使用方法

    swift 字符串String的使用方法

    這篇文章主要介紹了swift 字符串String的使用方法的相關資料,需要的朋友可以參考下
    2017-06-06
  • 理解二叉堆數(shù)據(jù)結(jié)構(gòu)及Swift的堆排序算法實現(xiàn)示例

    理解二叉堆數(shù)據(jù)結(jié)構(gòu)及Swift的堆排序算法實現(xiàn)示例

    二插堆即是完全二叉樹,對于排序可以按構(gòu)建最大堆或最小堆的方式來實現(xiàn),這里我們就來共同理解二叉堆數(shù)據(jù)結(jié)構(gòu)及Swift的堆排序算法實現(xiàn)示例
    2016-07-07
  • iOS開發(fā)中Swift 指紋驗證功能模塊實例代碼

    iOS開發(fā)中Swift 指紋驗證功能模塊實例代碼

    本文給大家分享ios調(diào)用touchid代碼塊,非常不錯,具有參考借鑒價值,需要的朋友參考下把
    2017-03-03
  • Swift?中?Opaque?Types學習指南

    Swift?中?Opaque?Types學習指南

    這篇文章主要為大家介紹了Swift?中?Opaque?Types學習指南,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04
  • 深入解析Swift中switch語句對case的數(shù)據(jù)類型匹配的支持

    深入解析Swift中switch語句對case的數(shù)據(jù)類型匹配的支持

    這篇文章主要介紹了Swift中switch語句對case的數(shù)據(jù)類型匹配的支持,Swift中switch...case語句支持多種數(shù)據(jù)類型的匹配判斷,十分強大,需要的朋友可以參考下
    2016-04-04
  • swift內(nèi)存管理指針類型使用實例詳解

    swift內(nèi)存管理指針類型使用實例詳解

    這篇文章主要為大家介紹了swift內(nèi)存管理指針類型使用實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11

最新評論