Swift函數(shù)提前返回實(shí)例詳解
簡評:
函數(shù)是一個(gè)組織在一起語句集合,以執(zhí)行特定任務(wù)。Swift 函數(shù)類似于簡單 C 函數(shù)以及復(fù)雜的 Objective C 語言函數(shù)。 它使我們能夠通過函數(shù)調(diào)用內(nèi)部的局部和全局參數(shù)值。 像其他任何語言一樣 swift 函數(shù)也遵循相同的步驟。
- 函數(shù)聲明:它告訴編譯器有關(guān)的函數(shù)的名稱,返回類型和參數(shù)。
- 函數(shù)定義:它提供函數(shù)的實(shí)際主體。
Swift 函數(shù)包含參數(shù)類型和返回類型。
函數(shù)提前返回主要的好處是:將每個(gè)錯(cuò)誤處理進(jìn)行分離,審查代碼時(shí)不需要考慮多種復(fù)雜異常,我們可以吧注意力集中在也業(yè)務(wù)邏輯中,調(diào)試代碼時(shí)可以直接在異常中打斷點(diǎn)。
提前返回
首先來看一下需要改進(jìn)的代碼示例,我們構(gòu)建一個(gè)筆記應(yīng)用使用 NotificationCenter API,當(dāng)筆記內(nèi)容有變化時(shí) Notification 來通知筆記列表變更,代碼如下:
class NoteListViewController: UIViewController { @objc func handleChangeNotification(_ notification: Notification) { let noteInfo = notification.userInfo?["note"] as? [String : Any] if let id = noteInfo?["id"] as? Int { if let note = database.loadNote(withID: id) { notes[id] = note tableView.reloadData() } } } }
上面的代碼可以很好的工作,但是可讀性差了點(diǎn)。因?yàn)檫@段代碼包含多重縮進(jìn)和類型轉(zhuǎn)換。我們來嘗試改進(jìn)這段代碼。
- 將方法提前返回,讓我們函數(shù)盡可能的快的返回。
- 使用 guard 替代 if,以避免嵌套。
class NoteListViewController: UIViewController { @objc func handleChangeNotification(_ notification: Notification) { let noteInfo = notification.userInfo?["note"] as? [String : Any] guard let id = noteInfo?["id"] as? Int else { return } guard let note = database.loadNote(withID: id) else { return } notes[id] = note tableView.reloadData() } }
將函數(shù)提前返回能夠?qū)⒐δ苁〉那闆r處理得更加清晰,這不僅提高了可讀性(更少的縮進(jìn),更少的嵌套),同時(shí)也有利于單元測試。
我們可以進(jìn)一步改進(jìn)代碼,將獲取 noteID 和類型轉(zhuǎn)換的代碼放在 Notification Extension 中,這樣就將 handleChangeNotification 業(yè)務(wù)邏輯和具體細(xì)節(jié)分離開來。修改后代碼如下所示:
private extension Notification { var noteID: Int? { let info = userInfo?["note"] as? [String : Any] return info?["id"] as? Int } } class NoteListViewController: UIViewController { @objc func handleChangeNotification(_ notification: Notification) { guard let id = notification.noteID else { return } guard let note = database.loadNote(withID: id) else { return } notes[id] = note tableView.reloadData() } }
這種結(jié)構(gòu)還大大簡化了調(diào)試的難度,我們可以直接在每個(gè) guard 中 return 中添加斷點(diǎn)來截獲所有失敗情況,而不需要單步執(zhí)行所有邏輯。
條件構(gòu)造
當(dāng)構(gòu)造一個(gè)對象實(shí)例,非常普遍的需求是需要構(gòu)建哪類對象取決于一系列的條件。
例如,啟動(dòng)應(yīng)用程序時(shí)顯示哪個(gè) view controller 取決于:
- 是否已經(jīng)登錄。
- 用戶是否已經(jīng)完成入職流程(onboarding flow)。
我們對這些條件的的實(shí)現(xiàn)可能是一系列的 if 和 else 語句,如下所示:
func showInitialViewController() { if loginManager.isUserLoggedIn { if tutorialManager.isOnboardingCompleted { navigationController.viewControllers = [HomeViewController()] } else { navigationController.viewControllers = [OnboardingViewController()] } } else { navigationController.viewControllers = [LoginViewController()] } }
同樣的提前返回和 guard 語句可以提升代碼可讀性,但是現(xiàn)在這種情況不是處理失敗情況,而是在不同條件下構(gòu)建不同 view controller。
現(xiàn)在來改進(jìn)這段代碼,使用輕量級的工程模式,將構(gòu)造初始界面移動(dòng)到專門的函數(shù)中,該函數(shù)返回匹配條件的view controller。如下所示:
func makeInitialViewController() -> UIViewController { guard loginManager.isUserLoggedIn else { return LoginViewController() } guard tutorialManager.isOnboardingCompleted else { return OnboardingViewController() } return HomeViewController() } func showInitialViewController() { let viewController = makeInitialViewController() navigationController.viewControllers = [viewController] }
由于 makeInitialViewController 方法是個(gè)純函數(shù)(不影響外部狀態(tài),固定輸入能夠得到固定輸出),實(shí)際上影響外部狀態(tài)的只有一個(gè)地方 navigationController.viewControllers = [viewController] ,(在日常開發(fā)中狀態(tài)如果沒有得到很好的控制很容易引起 bug,所以使用更少狀態(tài)和減少對狀態(tài)的修改可以一定程度上減少 bug 出現(xiàn)的幾率)。
條件控制
最后我們來看看,函數(shù)如何簡化復(fù)雜的條件邏輯。我們來構(gòu)建一個(gè) view controller 來顯示社交應(yīng)用的評論功能,如果滿足三個(gè)條件則運(yùn)行用戶對評論進(jìn)行編輯。代碼如下:
class CommentViewController: UIViewController { override func viewDidLoad() { super.viewDidLoad() if comment.authorID == user.id { if comment.replies.isEmpty { if !comment.edited { let editButton = UIButton() ... view.addSubview(editButton) } } } ... } }
這里使用了 3 個(gè) if 嵌套邏輯,每次重新審查代碼都會(huì)比較困擾,更具之前的經(jīng)驗(yàn)我們可以對代碼進(jìn)行優(yōu)化,添加 Comment extension:
extension Comment { func canBeEdited(by user: User) -> Bool { guard authorID == user.id else { return false } guard comment.replies.isEmpty else { return false } return !edited } } class CommentViewController: UIViewController { override func viewDidLoad() { super.viewDidLoad() if comment.canBeEdited(by: user) { let editButton = UIButton() ... view.addSubview(editButton) } ... } }
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
利用Swift實(shí)現(xiàn)各類的CATransition動(dòng)畫詳解
CATransition動(dòng)畫主要在過渡時(shí)使用,比如兩個(gè)頁面層級改變的時(shí)候添加一個(gè)轉(zhuǎn)場效果。CATransition分為兩類,一類是公開的動(dòng)畫效果,一類是非公開的動(dòng)畫效果。這篇文章主要給大家介紹了關(guān)于如何利用Swift實(shí)現(xiàn)各類CATransition動(dòng)畫的相關(guān)資料,需要的朋友可以參考下。2017-09-09在SpringBoot中實(shí)現(xiàn)適配器模式的兩種方式
這篇文章主要介紹了在SpringBoot中實(shí)現(xiàn)適配器模式的兩種方式,通過實(shí)現(xiàn)類定義類型字段實(shí)現(xiàn)和以動(dòng)態(tài)service名稱的方式實(shí)現(xiàn),并且通過代碼示例講解的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2024-03-03Swift利用CoreData如何存儲(chǔ)多種數(shù)據(jù)類的通訊錄
這篇文章主要給大家介紹了關(guān)于Swift利用CoreData如何存儲(chǔ)多種數(shù)據(jù)類的通訊錄的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12Swift實(shí)現(xiàn)復(fù)數(shù)計(jì)算器
這篇文章主要為大家詳細(xì)介紹了Swift實(shí)現(xiàn)復(fù)數(shù)計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01Swift解決UITableView空數(shù)據(jù)視圖問題的簡單方法
這篇文章主要給大家介紹了關(guān)于Swift解決UITableView空數(shù)據(jù)視圖問題的簡單方法,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用swift具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2018-10-10