文章出處

轉眼間,Swift已經一歲多了,這門新鮮、語法時尚、類型安全、執行速度更快的語言已經漸漸的深入廣大開發者的心。

今年6月,一年一度的WWDC大會如期而至,在大會上Apple發布了Swift 2.0,引入了很多新的特性,以幫助開發者能更快,更簡單的構建應用。我在這里也說道說道Swift 2.0中值得大家注意的新特性。

guard語句

guard語句和if語句有點類似,都是根據其關鍵字之后的表達式的布爾值決定下一步執行什么。但與if語句不同的是,guard語句只會有一個代碼塊,不像if語句可以if else多個代碼塊。

那么guard語句的作用到底是什么呢?顧名思義,就是守護。guard語句判斷其后的表達式布爾值為false時,才會執行之后代碼塊里的代碼,如果為true,則跳過整個guard語句,我們舉例來看看。

我們以今年高考為例,在進入考場時一般都會檢查身份證和準考證,我們寫這樣一個方法:

func checkup(person: [String: String!]) {
  
    // 檢查身份證,如果身份證沒帶,則不能進入考場
    guard let id = person["id"] else {
        print("沒有身份證,不能進入考場!")
        return
    }
    
    // 檢查準考證,如果準考證沒帶,則不能進入考場
    guard let examNumber = person["examNumber"] else {
        print("沒有準考證,不能進入考場!")
        return
    }
    
    // 身份證和準考證齊全,方可進入考場
    print("您的身份證號為:\(id),準考證號為:\(examNumber)。請進入考場!")
    
}

checkup(["id": "123456"]) // 沒有準考證,不能進入考場!
checkup(["examNumber": "654321"]) // 沒有身份證,不能進入考場!
checkup(["id": "123456", "examNumber": "654321"]) // 您的身份證號為:123456,準考證號為:654321。請進入考場!

上述代碼中的第一個guard語句用于檢查身份證,如果檢查到身份證沒帶,也就是表達式為false時,執行大括號里的代碼,并返回。第二個guard語句則檢查準考證。

如果兩證齊全,則執行最后一個打印語句,上面的兩個guard語句大括號內的代碼都不會執行,因為他們表達式的布爾值都是true

這里值得注意的是,idexamNumber可以在guard語句之外使用,也就是說當guard對其表達式進行驗證后,idexamNumber可在整個方法的作用域中使用,并且是解包后的。

我們再用if else語句寫一個類似的方法:

func checkupUseIf(person: [String: String!]) {
    
    if let id = person["id"], let examNumber = person["examNumber"] {
        print("您的身份證號為:\(id),準考證號為:\(examNumber)。請進入考場!")
    } else {
        print("證件不齊全,不能進入考場!")
    }
    
    print("您的身份證號為:\(id),準考證號為:\(examNumber)")  // 報異常
    
}

checkupUseIf(["id": "123456"]) // 證件不齊全,不能進入考場!
checkupUseIf(["examNumber": "654321"]) // 證件不齊全,不能進入考場!
checkupUseIf(["id": "123456", "examNumber": "654321"]) // 您的身份證號為:123456,準考證號為:654321。請進入考場!

我們可以看到用if else實現的方法顯然不如guard實現的那么精準。而且idexamNumber的作用域只限在if的第一個大括號內,超出這個作用域編譯就會報錯。

通過上述兩個小例子不難看出,guard語句正如一個稱職的守衛,層層把關,嚴防一切不允許發生的事,并且讓代碼具有更高的可讀性,非常棒。

異常處理

在Swift 1.0時代是沒有異常處理和拋出機制的,如果要處理異常,要么使用if else語句或switch語句判斷處理,要么使用閉包形式的回調函數處理,再要么就使用NSError處理。以上這些方法都不能像Java中的try catch異常控制語句那樣行如流水、從容不迫的處理異常,而且也會降低代碼的可讀性。當Swift 2.0到來后,一切都不一樣了。

在Swift 2.0中Apple提供了使用throwsthrowtrydocatch這五個關鍵字組成的異常控制處理機制。下面我們來舉例看看如何使用,我用使用手機刷朋友圈為例。

首先我們需要定義異常枚舉,在Swift 2.0中Apple提供了ErrorType協議需要我們自定義的異常枚舉遵循:

enum WechatError: ErrorType {
    case NoBattery // 手機沒電
    case NoNetwork // 手機沒網
    case NoDataStream // 手機沒有流量
}

我們定義了導致不能刷微信的錯誤枚舉’wechatError。然后定義一個檢查是否可以刷微信的方法checkIsWechatOk()

func checkIsWechatOk(isPhoneHasBattery: Bool, isPhoneHasNetwork: Bool, dataStream: Int) throws {
    
    guard isPhoneHasBattery else {
        throw WechatError.NoBattery
    }
    
    guard isPhoneHasNetwork else {
        throw WechatError.NoNetwork
    }
    
    guard dataStream > 50 else {
        throw WechatError.NoDataStream
    }
    
}

這里注意,在方法名后有throws關鍵字,意思為該方法產生的異常向上層拋出。在方法體內使用guard語句對各種狀態進行判斷,然后使用throw關鍵字拋出對應的異常。然后我們定義刷微信的方法:

func playWechat(isPhoneHasBattery: Bool, isPhoneHasNetwork: Bool, dataStream: Int) {
    
    do {
        try checkIsWechatOk(isPhoneHasBattery, isPhoneHasNetwork: isPhoneHasNetwork, dataStream: dataStream)
        print("放心刷,刷到天昏地暗!")
    } catch WechatError.NoBattery {
        print("手機都沒電,刷個鬼啊!")
    } catch WechatError.NoNetwork {
        print("沒有網絡哎,洗洗玩單機吧!")
    } catch WechatError.NoDataStream {
        print("沒有流量了,去蹭Wifi吧!")
    } catch {
        print("見鬼了!")
    }
    
}

playWechat(true, isPhoneHasNetwork: true, dataStream: 60) // 放心刷,刷到天昏地暗!
playWechat(true, isPhoneHasNetwork: false, dataStream: 60) // 沒有網絡哎,洗洗玩單機吧!
playWechat
arrow
arrow
    全站熱搜
    創作者介紹
    創作者 大師兄 的頭像
    大師兄

    IT工程師數位筆記本

    大師兄 發表在 痞客邦 留言(0) 人氣()