swift中defer的實際應用小結(jié)
看看蘋果官方的介紹
用 defer 語句在即將離開當前代碼塊時執(zhí)行一系列語句。該語句讓你能執(zhí)行一些必要的清理工作,不管是以何種方式離開當前代碼塊的——無論是由于拋出錯誤而離開,或是由于諸如 return、break 的語句。例如,你可以用 defer 語句來確保文件描述符得以關閉,以及手動分配的內(nèi)存得以釋放。
defer 語句將代碼的執(zhí)行延遲到當前的作用域退出之前。該語句由 defer 關鍵字和要被延遲執(zhí)行的語句組成。延遲執(zhí)行的語句不能包含任何控制轉(zhuǎn)移語句,例如 break、return 語句,或是拋出一個錯誤。延遲執(zhí)行的操作會按照它們聲明的順序從后往前執(zhí)行——也就是說,第一條 defer 語句中的代碼最后才執(zhí)行,第二條 defer 語句中的代碼倒數(shù)第二個執(zhí)行,以此類推。最后一條語句會第一個執(zhí)行。
總結(jié)一下蘋果官方的介紹
defer語句在代碼塊(方法、閉包等,可以理解為大括號包裝起來的代碼)作用域退出之前\color{red}{作用域退出之前}作用域退出之前執(zhí)行,也就是代碼塊中其他應該執(zhí)行的代碼都執(zhí)行完了,才執(zhí)行defer中的代碼
一個代碼塊允許多個defer,多個defer執(zhí)行的順序從后到前\color{red}{從后到前}從后到前
一些測試及誤區(qū)糾正
測試案例1
func testDefer() { defer { print("方法中defer內(nèi)容") } if true { defer { print("if 中defer內(nèi)容") } print("if中最后的代碼") } print("方法中的代碼") if true { return } print("方法結(jié)束前最后一句代碼") } testDefer()
以上代碼打印結(jié)果:
if中最后的代碼
if 中defer內(nèi)容
方法中的代碼
方法中defer內(nèi)容
打印結(jié)果中,第一個if中的代碼及里面的defer最先執(zhí)行,方法中的defer最后執(zhí)行,由此可以看出,代碼塊中其他能夠執(zhí)行的代碼先執(zhí)行,最后執(zhí)行defer的內(nèi)容;defer的作用范圍不能簡單的看成方法,而是代碼塊(可能有些同學會有這樣的誤區(qū))
測試案例2
func testDefer() { print("開始") defer { print("defer 1 中的內(nèi)容") } defer { print("defer 2 中的內(nèi)容") } if true { return } defer { print("defer 3 中的內(nèi)容") } print("方法結(jié)束前最后一句代碼") } testDefer()
打印結(jié)果
開始
defer 2 中的內(nèi)容
defer 1 中的內(nèi)容
我們可以看到最后一個defer沒有執(zhí)行,所以defer定義的位置很重要,如果沒有執(zhí)行defer定義的代碼,在代碼塊結(jié)束前不會執(zhí)行defer中的內(nèi)容
多個defer的執(zhí)行順序從后到前
一些實際應用場景
場景1:一些資源用完后需釋放,這里給的是官方的一個案例
func processFile(filename: String) throws { if exists(filename) { let file = open(filename) defer { close(file) } while let line = try file.readline() { // 處理文件。 } // close(file) 會在這里被調(diào)用,即作用域的最后。 } }
開始用到資源的時候就使用defer去釋放,避免忘記釋放資源
場景2:加鎖解鎖,借鑒了kingfisher
let lock = NSLock() func testDefer() { lock.lock() defer { lock.unlock() } doSomething() } testDefer()
在加鎖后立刻用defer解鎖,避免忘記解鎖
場景3:處理一些代碼塊作用域結(jié)束前的重復操作,比如請求網(wǎng)絡數(shù)據(jù)的時候
通常的一種寫法
func loadCityList(_ finish: ((Error?, [String]?) -> ())?) { DispatchQueue.global().async { // 模擬網(wǎng)絡請求 let data: AnyObject? // 模擬服務器返回的數(shù)據(jù) guard let dict = data as? [String: AnyObject] else { DispatchQueue.main.async { finish?(error, nil) } return } guard let code = dict["code"] as? Int, code == 200 else { DispatchQueue.main.async { finish?(error, nil) } return } guard let citys = dict["data"] as? [String]? else { DispatchQueue.main.async { finish?(error, nil) } return } DispatchQueue.main.async { finish?(nil, citys) } } }
當每次有錯誤處理時和結(jié)果正確時都需要去做回調(diào),而且回調(diào)可能有一堆代碼,看起來代碼會比較冗余,而且在一些錯誤處理時很容易造成忘記回調(diào)
defer怎么去寫呢
func loadCityList(_ finish: ((Error?, [String]?) -> ())?) { DispatchQueue.global().async { // 模擬網(wǎng)絡請求 var error: Error? = nil var citys: [String]? = nil defer { DispatchQueue.main.async { finish?(error, citys) } } let data: AnyObject? // 模擬服務器返回的數(shù)據(jù) guard let dict = data as? [String: AnyObject] else { error = ... return } guard let code = dict["code"] as? Int, code == 200 else { error = ... return } guard let tempCitys = dict["data"] as? [String]? else { error = ... return } citys = tempCitys } }
使用defer既解決了代碼冗余,又解決了可能忘記回調(diào)的問題,還有當我們看到defer時,我們很清楚知道,無論網(wǎng)絡請求結(jié)果如果,都會回調(diào)
總結(jié)
本文主要介紹了defer的定義、作用及一些用法
到此這篇關于swift中defer的實際應用的文章就介紹到這了,更多相關swift中defer應用內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Swift中的類class與結(jié)構體struct體學習筆記
和C++一樣,Swfit中同時擁有類與結(jié)構體,能夠充分滿足開發(fā)者面向?qū)ο蠛兔嫦蜻^程編程的需求,這里我們就來看一下Swift中的類class與結(jié)構體struct體學習筆記2016-07-07Swift免費短信驗證碼實現(xiàn)及動態(tài)倒計時功能
這篇文章主要介紹了Swift免費短信驗證碼實現(xiàn)及動態(tài)倒計時功能的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-02-02