swift異常處理

一.異常的介紹
只要我們?cè)诰幊?,就一定要面?duì)錯(cuò)誤處理的問(wèn)題。
Swift在設(shè)計(jì)的時(shí)候就盡可能讓我們明確感知錯(cuò)誤,明確處理錯(cuò)誤
比如:只有使用Optional才能處理空值;
如何描述一個(gè)錯(cuò)誤?
在Swift里,任何一個(gè)遵從ErrorType protocol的類(lèi)型,都可以用于描述錯(cuò)誤。
ErrorType是一個(gè)空的protocol,它唯一的功能,就是告訴Swift編譯器,某個(gè)類(lèi)型用來(lái)表示一個(gè)錯(cuò)誤。
通常,我們使用一個(gè)enum來(lái)定義各種錯(cuò)誤的可能性
二.異常的示例
假如我們想要讀取一個(gè)文件中的內(nèi)容,按照OC的邏輯我們可以這樣來(lái)模擬
當(dāng)我們調(diào)用方法獲取結(jié)果為nil時(shí),你并不能確定到底參數(shù)了什么錯(cuò)誤得到了nil

func readFileContent(filePath : String) -> String? {
    // 1.filePath為
    if filePath == "" {
        return nil
    }
    // 2.filepath有值,但是沒(méi)有對(duì)應(yīng)的文件
    if filePath != "/User/Desktop/123.plist" {
        return nil
    }
    // 3.取出其中的內(nèi)容
    return "123"
}
readFileContent("abc")
 - 使用異常對(duì)上述方法進(jìn)行改進(jìn)
// 1.定義異常
enum FileReadError : ErrorType {
    case FileISNull
    case FileNotFound
}
// 2.改進(jìn)方法,讓方法拋出異常
func readFileContent(filePath : String) throws -> String {
    // 1.filePath為""
    if filePath == "" {
        throw FileReadError.FileISNull
    }
    // 2.filepath有值,但是沒(méi)有對(duì)應(yīng)的文件
    if filePath != "/User/Desktop/123.plist" {
        throw FileReadError.FileISNull
    }
    // 3.取出其中的內(nèi)容
    return "123"
}

三.處理異常有三種方式
3.異常的處理三種方式
3.1.try方式,需要手動(dòng)處理異常

do {
    let result = try readFileContent("abc")
} catch {
    print(error)
}

3.2.try?方式,不處理異常,如果出現(xiàn)了異常,則返回一個(gè)nil.沒(méi)有異常,則返回對(duì)應(yīng)的值
最終返回結(jié)果為一個(gè)可選類(lèi)型

let result = try? readFileContent("abc")

3.3.try!方法,告訴系統(tǒng)該方法沒(méi)有異常.
注意:如果出現(xiàn)了異常,則程序會(huì)崩潰

try! readFileContent("abc")

如何拋出異常
在拋出異常之前,我們需要在函數(shù)或方法的返回箭頭 -> 前使用 throws 來(lái)標(biāo)明將會(huì)拋出異常

func myMethodRetrunString() throws -> String

// No return, we can just add throws in the end
func myMethodRetrunNothing() throws

聲明之后, 我們需要在函數(shù)或者方法里扔出異常,很簡(jiǎn)單使用throw 就可以了

func myMethod() throws
  //...
  // item is an optional value
  guard let item = item else {
    // need throws the error out
    throw MyError.NotExist
  }
  // do with item
}

上面這段代碼使用了guard
來(lái)進(jìn)行unwrap optional value。這是 Swift 2.0 提供的一個(gè)新的方法。
Guard
在 Haskell, Erlang 等語(yǔ)言中早已存在guard
, 在這里有更多關(guān)于它的介紹。
guard
翻譯過(guò)來(lái)可以理解為守護(hù),守衛(wèi)。
在 Swift 中,guard
有點(diǎn)像if
但是他們有兩個(gè)非常重要的區(qū)別
guard
必須強(qiáng)制有else
語(yǔ)句
只有在guard
審查的條件成立,guard
之后的代碼才會(huì)運(yùn)行 (像守衛(wèi)一樣,條件不符就不讓過(guò)去)。

guard
中的else
只能執(zhí)行轉(zhuǎn)換語(yǔ)句,像return
,break
,continue
或者throws
當(dāng)然你也可以在這里返后一個(gè)函數(shù)或者方法。
值得注意的是,guard
的使用會(huì)提高你代碼的可讀性,但是也代表你的代碼的執(zhí)行會(huì)有非常明確的順序性,這一點(diǎn)需要開(kāi)發(fā)者們留心處理。
雖然我們?cè)诋惓L幚碇刑岬搅薵uard
但是不代表它只能在異常處理中使用。它具有廣泛的適用性,或許過(guò)陣子我會(huì)專(zhuān)門(mén)為guard
的使用寫(xiě)篇文章。
如何獲取并處理異常?
上文講述了如何建造拋出異常,獲取和處理異常就變得很簡(jiǎn)單了。使用do-catch
機(jī)制。
文/阮超(簡(jiǎn)書(shū)作者)原文鏈接:http://www.itdecent.cn/p/96a7db3fde00著作權(quán)歸作者所有,轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),并標(biāo)注“簡(jiǎn)書(shū)作者”。

 do {
    try functionWillThrowError()
  } catch {
    // deal with error
  }

do-catch
機(jī)制簡(jiǎn)單易懂。很多編程語(yǔ)言也使用類(lèi)似的機(jī)制進(jìn)行異常處理,但是在 Swift 中有一個(gè)比較重要的特性。
catch
和switch
一樣具有 Pattern Matching 的能力。所以,使用catch
你可以對(duì)異常的解析進(jìn)行更為高級(jí)的處理
文/阮超(簡(jiǎn)書(shū)作者)原文鏈接:http://www.itdecent.cn/p/96a7db3fde00著作權(quán)歸作者所有,轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),并標(biāo)注“簡(jiǎn)書(shū)作者”。

do {
    try functionWillThrowError()
  } catch MyError.NotExist {
    // deal with not exist
  } catch MyError.OutOfRange {
    // deal with not exist
  }

這里值得提一下在 Swift 2.0中一個(gè)跟異常處理沒(méi)有關(guān)系的改進(jìn)

Swift 2.0 中沒(méi)有了 do-while循環(huán),取而代之的是 repeat-while。蘋(píng)果說(shuō)這個(gè)改動(dòng)是為了增強(qiáng)代碼的可讀性。但是我更覺(jué)得是為了讓我們更舒服的使用 do-catch
不處理異常
如果我不想處理異常怎么辦,或者說(shuō),我非常確定某個(gè)方法或者函數(shù)雖然聲明會(huì)拋出異常,但是我自己知道我在使用時(shí)候是絕對(duì)不會(huì)拋出任何異常的。這種情況下 我們可以使用 try!

文/阮超(簡(jiǎn)書(shū)作者)
原文鏈接:http://www.itdecent.cn/p/96a7db3fde00
著作權(quán)歸作者所有,轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),并標(biāo)注“簡(jiǎn)書(shū)作者”。

 try! functionThrowErrorNil()

當(dāng)然,如果你使用 try!,而你的方法或者函數(shù)拋出了異常,那么你會(huì)得到一個(gè)運(yùn)行中異常 (runtime error) 所以我們開(kāi)發(fā)者需要慎用哦。

Defer
文章結(jié)束前我們?cè)儆懻撓?defer

在你的代碼塊就要結(jié)束前。如果你使用了 defer。 在其之中的代碼就會(huì)運(yùn)行。等于說(shuō),給了你最后的機(jī)會(huì)來(lái)進(jìn)行一些處理。如果你熟悉 BDD 或者 TDD, 那么你可以參考他們中的 aferAll 機(jī)制

文/阮超(簡(jiǎn)書(shū)作者)
原文鏈接:http://www.itdecent.cn/p/96a7db3fde00
著作權(quán)歸作者所有,轉(zhuǎn)載請(qǐng)聯(lián)系作者獲得授權(quán),并標(biāo)注“簡(jiǎn)書(shū)作者”。

func myFunction() throws {  
  defer {
    // No matter what happened I need do something
    print("All done, clean up here")
  }
  guard let item = item else {
    // need throws the error out
    throw MyError.NotExist
  }

  guard item.count > maxNumber else {
    // need throws the error out
    throw MyError.OutOfRange
  }

  // do something with item
  // ...
}

注意,如果你有多個(gè)defer 語(yǔ)句,他們?cè)趫?zhí)行的順序會(huì)和棧一樣,最后一個(gè)進(jìn),第一個(gè)出.
總結(jié)
使用 ErrorType 的幫助建立你的異常類(lèi)型
使用 throws 來(lái)聲明異常,用 throw 來(lái)拋出異常
使用 do-catch 機(jī)制來(lái)獲取和處理異常

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • swift像其他編程語(yǔ)言一樣可以?huà)伋霎惓2⑻幚懋惓?,本文就詳?xì)說(shuō)一下在swift中拋出并處理異常,swift中的一...
    初光夫閱讀 1,033評(píng)論 0 51
  • 序言 對(duì)異常的正確處理能夠明確反映在代碼執(zhí)行中出現(xiàn)的問(wèn)題,使得在接下來(lái)的debug快速定位問(wèn)題所在的地方,提高de...
    nuclear閱讀 812評(píng)論 0 1
  • 如何建造異常類(lèi)型? 在 iOS 開(kāi)發(fā)當(dāng)中,我們會(huì)面對(duì)很多異常處理。在 Cocoa Touch 中我們使用 NSEr...
    GoGooGooo閱讀 283評(píng)論 0 0
  • 我早該知道的,我們的關(guān)系,不過(guò)是你需要人陪伴的時(shí)候,我恰好在。 2017年11月19日 星期日 晴 文|深海夢(mèng)影 ...
    深海夢(mèng)影閱讀 4,159評(píng)論 78 155
  • /奶茶爸爸/ 某人在京東花79買(mǎi)到15就能買(mǎi)到的轉(zhuǎn)換頭且用一次就壞掉了,對(duì)植脂末產(chǎn)生了排斥反應(yīng)。 /勤儉持家/ 某...
    rukidin閱讀 266評(píng)論 0 0

友情鏈接更多精彩內(nèi)容