1. 錯(cuò)誤處理
錯(cuò)誤處理(Error handling)是響應(yīng)錯(cuò)誤以及從錯(cuò)誤中恢復(fù)的過(guò)程默怨。Swift 提供了在運(yùn)行時(shí)對(duì)可恢復(fù)錯(cuò)誤的拋出、捕獲骤素、傳遞和操作的一等公民支持匙睹。
1.1 表示并拋出錯(cuò)誤
在 Swift 中,錯(cuò)誤用符合Error 協(xié)議的類型的值來(lái)表示济竹。這個(gè)空協(xié)議表明該類型可以用于錯(cuò)誤處理痕檬。
Swift 的枚舉類型尤為適合構(gòu)建一組相關(guān)的錯(cuò)誤狀態(tài),枚舉的關(guān)聯(lián)值還可以提供錯(cuò)誤狀態(tài)的額外信息
enum VendingMachineError: Error {
case invalidSelection //選擇無(wú)效
case insufficientFunds(coinsNeeded: Int) //金額不足
case outOfStock //缺貨
}
1.2 處理錯(cuò)誤
某個(gè)錯(cuò)誤被拋出時(shí)送浊,附近的某部分代碼必須負(fù)責(zé)處理這個(gè)錯(cuò)誤梦谜,例如糾正這個(gè)問(wèn)題、嘗試另外一種方式袭景、或是向用戶報(bào)告錯(cuò)誤唁桩。
1.2.1 用 throwing 函數(shù)傳遞錯(cuò)誤
為了表示一個(gè)函數(shù)、方法或構(gòu)造器可以拋出錯(cuò)誤耸棒,在函數(shù)聲明的參數(shù)列表之后加上throws 關(guān)鍵字荒澡。一個(gè)標(biāo)有throws 關(guān)鍵字的函數(shù)被稱作throwing 函數(shù)。如果這個(gè)函數(shù)指明了返回值類型与殃, throws 關(guān)鍵詞需要寫在箭頭( -> )的前面单山。
func canThrowErrors() throws -> String
func cannotThrowErrors() -> String
一個(gè) throwing 函數(shù)可以在其內(nèi)部拋出錯(cuò)誤,并將錯(cuò)誤傳遞到函數(shù)被調(diào)用時(shí)的作用域幅疼。
注意
只有 throwing 函數(shù)可以傳遞錯(cuò)誤米奸。任何在某個(gè)非 throwing 函數(shù)內(nèi)部拋出的錯(cuò)誤只能在函數(shù)內(nèi)部處理。
struct Item {
var price: Int
var count: Int
}
class VendingMachine {
var inventory = [
"Candy Bar": Item(price: 12, count: 7),
"Chips": Item(price: 10, count: 4),
"Pretzels": Item(price: 7, count: 11)
]
var coinsDeposited = 0
func dispenseSnack(snack: String) {
print("Dispensing \(snack)")
}
func vend(itemNamed name: String) throws {
guard let item = inventory[name] else {
throw VendingMachineError.InvalidSelection
}
guard item.count > 0 else {
throw VendingMachineError.OutOfStock
}
guard item.price <= coinsDeposited else {
throw VendingMachineError.InsufficientFunds(coinsNeeded: item.price - coinsDeposited)
}
coinsDeposited -= item.price
var newItem = item
newItem.count -= 1
inventory[name] = newItem
print("Dispensing \(name)")
}
}
在vend(itemNamed:) 方法的實(shí)現(xiàn)中使用了guard 語(yǔ)句來(lái)提前退出方法爽篷,確保在購(gòu)買某個(gè)物品所需的條件中躏升,有任一條件不滿足時(shí),能提前退出方法并拋出相應(yīng)的錯(cuò)誤狼忱。由于throw 語(yǔ)句會(huì)立即退出方法膨疏,所以物品只有在所有條件都滿足時(shí)才會(huì)被售出一睁。
1.2.2 用 Do-Catch 處理錯(cuò)誤
可以使用一個(gè)do-catch 語(yǔ)句運(yùn)行一段閉包代碼來(lái)處理錯(cuò)誤。如果在do 子句中的代碼拋出了一個(gè)錯(cuò)誤佃却,這個(gè)錯(cuò)誤會(huì)與catch 子句做匹配者吁,從而決定哪條子句能處理它。
do {
try expression
statements
} catch pattern 1 {
statements
} catch pattern 2 where condition {
statements
}
1.2.3 將錯(cuò)誤轉(zhuǎn)換成可選值
可以使用try? 通過(guò)將錯(cuò)誤轉(zhuǎn)換成一個(gè)可選值來(lái)處理錯(cuò)誤饲帅。如果在評(píng)估try? 表達(dá)式時(shí)一個(gè)錯(cuò)誤被拋出复凳,那么表達(dá)式的值就是nil
func fetchData() -> Data? {
if let data = try? fetchDataFromDisk() { return data }
if let data = try? fetchDataFromServer() { return data }
return nil
}
1.2.4 禁用錯(cuò)誤傳遞
有時(shí)你知道某個(gè)throwing 函數(shù)實(shí)際上在運(yùn)行時(shí)是不會(huì)拋出錯(cuò)誤的,在這種情況下灶泵,你可以在表達(dá)式前面寫try! 來(lái)禁用錯(cuò)誤傳遞育八,這會(huì)把調(diào)用包裝在一個(gè)不會(huì)有錯(cuò)誤拋出的運(yùn)行時(shí)斷言中。如果真的拋出了錯(cuò)誤赦邻,你會(huì)得到一個(gè)運(yùn)行時(shí)錯(cuò)誤髓棋。
1.3 指定清理操作
可以使用defer 語(yǔ)句在即將離開當(dāng)前代碼塊時(shí)執(zhí)行一系列語(yǔ)句。該語(yǔ)句讓你能執(zhí)行一些必要的清理工作惶洲,不管是以何種方式離開當(dāng)前代碼塊的——無(wú)論是由于拋出錯(cuò)誤而離開按声,還是由于諸如return 或者break 的語(yǔ)句。例如恬吕,你可以用defer 語(yǔ)句來(lái)確保文件描述符得以關(guān)閉签则,以及手動(dòng)分配的內(nèi)存得以釋放。
defer 語(yǔ)句將代碼的執(zhí)行延遲到當(dāng)前的作用域退出之前铐料。該語(yǔ)句由defer 關(guān)鍵字和要被延遲執(zhí)行的語(yǔ)句組成渐裂。延遲執(zhí)行的語(yǔ)句不能包含任何控制轉(zhuǎn)移語(yǔ)句,例如break 或是return 語(yǔ)句钠惩,或是拋出一個(gè)錯(cuò)誤芯义。延遲執(zhí)行的操作會(huì)按照它們被指定時(shí)的順序的相反順序執(zhí)行——也就是說(shuō),第一條defer 語(yǔ)句中的代碼會(huì)在第二條defer 語(yǔ)句中的代碼被執(zhí)行之后才執(zhí)行妻柒,以此類推扛拨。
func processFile(filename: String) throws {
if exists(filename) {
let file = open(filename)
defer {
close(file)
}
while let line = try file.readline() {
// 處理文件。
}
// close(file) 會(huì)在這里被調(diào)用举塔,即作用域的最后绑警。
}
}
上面的代碼使用一條defer 語(yǔ)句來(lái)確保open(:) 函數(shù)有一個(gè)相應(yīng)的對(duì)close(:) 函數(shù)的調(diào)用。
注意
即使沒(méi)有涉及到錯(cuò)誤處理央渣,你也可以使用defer 語(yǔ)句计盒。
2. 類型轉(zhuǎn)換
<1>類型轉(zhuǎn)換可以判斷實(shí)例的類型,也可以將實(shí)例看做是其父類或者子類的實(shí)例芽丹。
<2>類型轉(zhuǎn)換在 Swift 中使用is 和as 操作符實(shí)現(xiàn)北启。這兩個(gè)操作符提供了一種簡(jiǎn)單達(dá)意的方式去檢查值的類型或者轉(zhuǎn)換它的類型。
<3>你也可以用它來(lái)檢查一個(gè)類型是否實(shí)現(xiàn)了某個(gè)協(xié)議
2.1 定義一個(gè)類層次作為例子
你可以將類型轉(zhuǎn)換用在類和子類的層次結(jié)構(gòu)上,檢查特定類實(shí)例的類型并且轉(zhuǎn)換這個(gè)類實(shí)例的類型成為這個(gè)層次結(jié)構(gòu)中的其他類型
2.2 檢查類型
用類型檢查操作符( is )來(lái)檢查一個(gè)實(shí)例是否屬于特定子類型咕村。若實(shí)例屬于那個(gè)子類型场钉,類型檢查操作符返回true ,否則返回false 懈涛。
var movieCount = 0
var songCount = 0
for item in library {
if item is Movie {
movieCount += 1
} else if item is Song {
songCount += 1
}
}
print("Media library contains \(movieCount) movies and \(songCount) songs")
// 打印 “Media library contains 2 movies and 3 songs
2.3 向下轉(zhuǎn)型
某類型的一個(gè)常量或變量可能在幕后實(shí)際上屬于一個(gè)子類逛万。當(dāng)確定是這種情況時(shí),你可以嘗試向下轉(zhuǎn)到它的子類型批钠,用類型轉(zhuǎn)換操作符( as? 或as! )宇植。
for item in library {
if let movie = item as? Movie {
print("Movie: '\(movie.name)', dir. \(movie.director)")
} else if let song = item as? Song {
print("Song: '\(song.name)', by \(song.artist)")
}
}
// Movie: 'Casablanca', dir. Michael Curtiz
注意
轉(zhuǎn)換沒(méi)有真的改變實(shí)例或它的值。根本的實(shí)例保持不變埋心;只是簡(jiǎn)單地把它作為它被轉(zhuǎn)換成的類型來(lái)使用指郁。
2.4 Any 和 AnyObject 的類型轉(zhuǎn)換
Swift 為不確定類型提供了兩種特殊的類型別名:
? Any 可以表示任何類型,包括函數(shù)類型拷呆。
? AnyObject 可以表示任何類類型的實(shí)例闲坎。
注意
Any 類型可以表示所有類型的值,包括可選類型洋腮。Swift 會(huì)在你用Any 類型來(lái)表示一個(gè)可選值的時(shí)候,給你一個(gè)警告手形。如果你確實(shí)想使用Any 類型來(lái)承載可選值啥供,你可以使用as 操作符顯示轉(zhuǎn)換為Any ,如下所示:
let optionalNumber: Int? = 3
things.append(optionalNumber) // 警告
things.append(optionalNumber as Any) // 沒(méi)有警告
3. 嵌套類型
枚舉常被用于為特定類或結(jié)構(gòu)體實(shí)現(xiàn)某些功能库糠。類似的伙狐,枚舉可以方便的定義工具類或結(jié)構(gòu)體,從而為某個(gè)復(fù)雜的類型所使用瞬欧。為了實(shí)現(xiàn)這種功能贷屎,Swift 允許你定義嵌套類型,可以在支持的類型中定義嵌套的枚舉艘虎、類和結(jié)構(gòu)體唉侄。
3.1 嵌套類型實(shí)踐
要在一個(gè)類型中嵌套另一個(gè)類型,將嵌套類型的定義寫在其外部類型的{} 內(nèi)野建,而且可以根據(jù)需要定義多級(jí)嵌套属划。
3.2 引用嵌套類型
在外部引用嵌套類型時(shí),在嵌套類型的類型名前加上其外部類型的類型名作為前綴:
let heartsSymbol = BlackjackCard.Suit.Hearts.rawValue
// 紅心符號(hào)為 “?”