5.控制流[Control Flow]

[The Swift Programming Language 中文版]
本頁包含內(nèi)容:

For-In 循環(huán)
While 循環(huán)
條件語句
控制轉(zhuǎn)移語句(Control Transfer Statements)
提前退出
檢測 API 可用性

Swift提供了多種流程控制結(jié)構(gòu)较锡,包括可以多次執(zhí)行任務(wù)的while循環(huán)娩缰,基于特定條件選擇執(zhí)行不同代碼分支的if勾效、guard和switch語句,還有控制流程跳轉(zhuǎn)到其他代碼的break和continue語句。

Swift 還增加了for-in循環(huán)胁住,用來更簡單地遍歷數(shù)組(array),字典(dictionary)刊咳,區(qū)間(range)彪见,字符串(string)和其他序列類型。

Swift 的switch語句比 C 語言中更加強大娱挨。在 C 語言中余指,如果某個 case 不小心漏寫了break,這個 case 就會貫穿至下一個 case让蕾,Swift 無需寫break浪规,所以不會發(fā)生這種貫穿的情況或听。case 還可以匹配更多的類型模式,包括區(qū)間匹配(range matching)笋婿,元組(tuple)和特定類型的描述誉裆。switch的 case 語句中匹配的值可以是由 case 體內(nèi)部臨時的常量或者變量決定,也可以由where分句描述更復(fù)雜的匹配條件缸濒。

For-In 循環(huán)
你可以使用for-in循環(huán)來遍歷一個集合里面的所有元素足丢,例如由數(shù)字表示的區(qū)間、數(shù)組中的元素庇配、字符串中的字符斩跌。

下面的例子用來輸出乘 5 乘法表前面一部分內(nèi)容:

for index in 1...5 {
    print("\(index) times 5 is \(index * 5)")
}
// 1 times 5 is 5
// 2 times 5 is 10
// 3 times 5 is 15
// 4 times 5 is 20
// 5 times 5 is 25

例子中用來進(jìn)行遍歷的元素是一組使用閉區(qū)間操作符(...)表示的從1到5的數(shù)字。index被賦值為閉區(qū)間中的第一個數(shù)字(1)捞慌,然后循環(huán)中的語句被執(zhí)行一次耀鸦。在本例中,這個循環(huán)只包含一個語句啸澡,用來輸出當(dāng)前index值所對應(yīng)的乘 5 乘法表結(jié)果袖订。該語句執(zhí)行后,index的值被更新為閉區(qū)間中的第二個數(shù)字(2)嗅虏,之后print(_:separator:terminator:)函數(shù)會再執(zhí)行一次洛姑。整個過程會進(jìn)行到閉區(qū)間結(jié)尾為止。

上面的例子中皮服,index是一個每次循環(huán)遍歷開始時被自動賦值的常量楞艾。這種情況下,index在使用前不需要聲明龄广,只需要將它包含在循環(huán)的聲明中硫眯,就可以對其進(jìn)行隱式聲明,而無需使用let關(guān)鍵字聲明蜀细。

如果你不需要知道區(qū)間序列內(nèi)每一項的值舟铜,你可以使用下劃線(_)替代變量名來忽略對值的訪問:

let base = 3
let power = 10
var answer = 1
for _ in 1...power {
    answer *= base
}
print("\(base) to the power of \(power) is \(answer)")
// 輸出 "3 to the power of 10 is 59049"

這個例子計算 base 這個數(shù)的 power 次冪(本例中,是3的10次冪)奠衔,從1(3的0次冪)開始做3的乘法谆刨, 進(jìn)行10次,使用1到10的閉區(qū)間循環(huán)归斤。這個計算并不需要知道每一次循環(huán)中計數(shù)器具體的值痊夭,只需要執(zhí)行了正確的循環(huán)次數(shù)即可。下劃線符號_(替代循環(huán)中的變量)能夠忽略具體的值脏里,并且不提供循環(huán)遍歷時對值的訪問她我。

使用for-in遍歷一個數(shù)組所有元素:

let names = ["Anna", "Alex", "Brian", "Jack"]
for name in names {
    print("Hello, \(name)!")
}
// Hello, Anna!
// Hello, Alex!
// Hello, Brian!
// Hello, Jack!

你也可以通過遍歷一個字典來訪問它的鍵值對。遍歷字典時,字典的每項元素會以(key, value)元組的形式返回番舆,你可以在for-in循環(huán)中使用顯式的常量名稱來解讀(key, value)元組酝碳。下面的例子中,字典的鍵(key)解讀為常量animalName恨狈,字典的值會被解讀為常量legCount:

let numberOfLegs = ["spider": 8, "ant": 6, "cat": 4]
for (animalName, legCount) in numberOfLegs {
    print("\(animalName)s have \(legCount) legs")
}
// ants have 6 legs
// cats have 4 legs
// spiders have 8 legs

字典元素的遍歷順序和插入順序可能不同疏哗,字典的內(nèi)容在內(nèi)部是無序的,所以遍歷元素時不能保證順序禾怠。關(guān)于數(shù)組和字典返奉,詳情參見集合類型。

While 循環(huán)
while循環(huán)運行一系列語句直到條件變成false吗氏。這類循環(huán)適合使用在第一次迭代前迭代次數(shù)未知的情況下芽偏。Swift 提供兩種while循環(huán)形式:

while循環(huán),每次在循環(huán)開始時計算條件是否符合弦讽;
repeat-while循環(huán)污尉,每次在循環(huán)結(jié)束時計算條件是否符合。

While

while循環(huán)從計算單一條件開始往产。如果條件為true十厢,會重復(fù)運行一系列語句,直到條件變?yōu)閒alse捂齐。

下面是一般情況下 while 循環(huán)格式:

while condition {  
    statements
}

下面的例子來玩一個叫做蛇和梯子的小游戲,也叫做滑道和梯子:


snakesAndLadders.png

游戲的規(guī)則如下:

游戲盤面包括 25 個方格缩抡,游戲目標(biāo)是達(dá)到或者超過第 25 個方格奠宜;
每一輪,你通過擲一個 6 邊的骰子來確定你移動方塊的步數(shù)瞻想,移動的路線由上圖中橫向的虛線所示压真;
如果在某輪結(jié)束,你移動到了梯子的底部蘑险,可以順著梯子爬上去滴肿;
如果在某輪結(jié)束,你移動到了蛇的頭部佃迄,你會順著蛇的身體滑下去泼差。
游戲盤面可以使用一個Int數(shù)組來表達(dá)。數(shù)組的長度由一個finalSquare常量儲存呵俏,用來初始化數(shù)組和檢測最終勝利條件堆缘。游戲盤面由 26 個 Int 0 值初始化,而不是 25 個(由0到25普碎,一共 26 個):

let finalSquare = 25
var board = [Int](count: finalSquare + 1, repeatedValue: 0)

一些方塊被設(shè)置成有蛇或者梯子的指定值吼肥。梯子底部的方塊是一個正值,使你可以向上移動,蛇頭處的方塊是一個負(fù)值缀皱,會讓你向下移動:

board[03] = +08; board[06] = +11; board[09] = +09; board[10] = +02
board[14] = -10; board[19] = -11; board[22] = -02; board[24] = -08

3 號方塊是梯子的底部斗这,會讓你向上移動到 11 號方格,我們使用board[03]等于+08(來表示11和3之間的差值)啤斗。使用一元加運算符(+i)是為了和一元減運算符(-i)對稱表箭,為了讓盤面代碼整齊,小于 10 的數(shù)字都使用 0 補齊(這些風(fēng)格上的調(diào)整都不是必須的争占,只是為了讓代碼看起來更加整潔)燃逻。

玩家由左下角編號為 0 的方格開始游戲。一般來說玩家第一次擲骰子后才會進(jìn)入游戲盤面:

var square = 0
var diceRoll = 0
while square < finalSquare {
    // 擲骰子
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    // 根據(jù)點數(shù)移動
    square += diceRoll
    if square < board.count {
        // 如果玩家還在棋盤上臂痕,順著梯子爬上去或者順著蛇滑下去
        square += board[square]
    }
}
print("Game over!")

本例中使用了最簡單的方法來模擬擲骰子伯襟。 diceRoll的值并不是一個隨機數(shù),而是以0為初始值握童,之后每一次while循環(huán)姆怪,diceRoll的值使用前置自增操作符(++i)來自增 1 ,然后檢測是否超出了最大值澡绩。任何時候如果diceRoll的值等于 7 時稽揭,就超過了骰子的最大值,會被重置為1肥卡。所以diceRoll的取值順序會一直是1溪掀。因此,diceRoll 所有的值只可能是 1 步鉴,2揪胃,3,4氛琢,5喊递,6,1阳似,2 等骚勘。

擲完骰子后,玩家向前移動diceRoll個方格撮奏,如果玩家移動超過了第 25 個方格俏讹,這個時候游戲結(jié)束,相應(yīng)地畜吊,代碼會在square增加board[square]的值向前或向后移動(遇到了梯子或者蛇)之前藐石,檢測square的值是否小于board的count屬性。

注意: 如果沒有這個檢測(square < board.count)定拟,board[square]可能會越界訪問board數(shù)組于微,導(dǎo)致錯誤逗嫡。例如如果square等于26, 代碼會去嘗試訪問board[26]株依,超過數(shù)組的長度驱证。
當(dāng)本輪while循環(huán)運行完畢,會再檢測循環(huán)條件是否需要再運行一次循環(huán)恋腕。如果玩家移動到或者超過第 25 個方格抹锄,循環(huán)條件結(jié)果為false,此時游戲結(jié)束荠藤。

while 循環(huán)比較適合本例中的這種情況伙单,因為在 while 循環(huán)開始時,我們并不知道游戲的長度或者循環(huán)的次數(shù)哈肖,只有在達(dá)成指定條件時循環(huán)才會結(jié)束吻育。

Repeat-While

while循環(huán)的另外一種形式是repeat-while,它和while的區(qū)別是在判斷循環(huán)條件之前淤井,先執(zhí)行一次循環(huán)的代碼塊布疼,然后重復(fù)循環(huán)直到條件為false。

注意: Swift語言的repeat-while循環(huán)合其他語言中的do-while循環(huán)是類似的币狠。
下面是一般情況下 repeat-while循環(huán)的格式:

repeat {
    statements
} while condition

還是蛇和梯子的游戲游两,使用repeat-while循環(huán)來替代while循環(huán)。finalSquare漩绵、board贱案、square和diceRoll的值初始化同while循環(huán)一樣:

let finalSquare = 25
var board = [Int](count: finalSquare + 1, repeatedValue: 0)
board[03] = +08; board[06] = +11; board[09] = +09; board[10] = +02
board[14] = -10; board[19] = -11; board[22] = -02; board[24] = -08
var square = 0
var diceRoll = 0

repeat-while的循環(huán)版本,循環(huán)中第一步就需要去檢測是否在梯子或者蛇的方塊上止吐。沒有梯子會讓玩家直接上到第 25 個方格轰坊,所以玩家不會通過梯子直接贏得游戲钦讳。這樣在循環(huán)開始時先檢測是否踩在梯子或者蛇上是安全的。

游戲開始時蠕嫁,玩家在第 0 個方格上估蹄,board[0]一直等于 0, 不會有什么影響:

repeat {
    // 順著梯子爬上去或者順著蛇滑下去
    square += board[square]
    // 擲骰子
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    // 根據(jù)點數(shù)移動
    square += diceRoll
} while square < finalSquare
print("Game over!")

檢測完玩家是否踩在梯子或者蛇上之后哈误,開始擲骰子,然后玩家向前移動diceRoll個方格,本輪循環(huán)結(jié)束套鹅。

循環(huán)條件(while square < finalSquare)和while方式相同,但是只會在循環(huán)結(jié)束后進(jìn)行計算汰具。在這個游戲中卓鹿,repeat-while表現(xiàn)得比while循環(huán)更好。repeat-while方式會在條件判斷square沒有超出后直接運行square += board[square]留荔,這種方式可以去掉while版本中的數(shù)組越界判斷吟孙。

條件語句
根據(jù)特定的條件執(zhí)行特定的代碼通常是十分有用的,例如:當(dāng)錯誤發(fā)生時,你可能想運行額外的代碼杰妓;或者藻治,當(dāng)輸入的值太大或太小時,向用戶顯示一條消息等巷挥。要實現(xiàn)這些功能桩卵,你就需要使用條件語句。

Swift 提供兩種類型的條件語句:if語句和switch語句倍宾。通常雏节,當(dāng)條件較為簡單且可能的情況很少時,使用if語句高职。而switch語句更適用于條件較復(fù)雜钩乍、可能情況較多且需要用到模式匹配(pattern-matching)的情境。

If

if語句最簡單的形式就是只包含一個條件初厚,當(dāng)且僅當(dāng)該條件為true時件蚕,才執(zhí)行相關(guān)代碼:

var temperatureInFahrenheit = 30
if temperatureInFahrenheit <= 32 {
    print("It's very cold. Consider wearing a scarf.")
}
// 輸出 "It's very cold. Consider wearing a scarf."

上面的例子會判斷溫度是否小于等于 32 華氏度(水的冰點)。如果是产禾,則打印一條消息排作;否則,不打印任何消息亚情,繼續(xù)執(zhí)行if塊后面的代碼妄痪。

當(dāng)然,if語句允許二選一楞件,也就是當(dāng)條件為false時衫生,執(zhí)行 else 語句:

temperatureInFahrenheit = 40
if temperatureInFahrenheit <= 32 {
    print("It's very cold. Consider wearing a scarf.")
} else {
    print("It's not that cold. Wear a t-shirt.")
}
// 輸出 "It's not that cold. Wear a t-shirt."

顯然,這兩條分支中總有一條會被執(zhí)行土浸。由于溫度已升至 40 華氏度罪针,不算太冷,沒必要再圍圍巾——因此黄伊,else分支就被觸發(fā)了泪酱。

你可以把多個if語句鏈接在一起,像下面這樣:

temperatureInFahrenheit = 90
if temperatureInFahrenheit <= 32 {
    print("It's very cold. Consider wearing a scarf.")
} else if temperatureInFahrenheit >= 86 {
    print("It's really warm. Don't forget to wear sunscreen.")
} else {
    print("It's not that cold. Wear a t-shirt.")
}
// 輸出 "It's really warm. Don't forget to wear sunscreen."

在上面的例子中还最,額外的if語句用于判斷是不是特別熱墓阀。而最后的else語句被保留了下來,用于打印既不冷也不熱時的消息拓轻。

實際上斯撮,最后的else語句是可選的:

temperatureInFahrenheit = 72
if temperatureInFahrenheit <= 32 {
    print("It's very cold. Consider wearing a scarf.")
} else if temperatureInFahrenheit >= 86 {
    print("It's really warm. Don't forget to wear sunscreen.")
}

在這個例子中,由于既不冷也不熱扶叉,所以不會觸發(fā)if或else if分支勿锅,也就不會打印任何消息帕膜。

Switch

switch語句會嘗試把某個值與若干個模式(pattern)進(jìn)行匹配。根據(jù)第一個匹配成功的模式粱甫,switch語句會執(zhí)行對應(yīng)的代碼泳叠。當(dāng)有可能的情況較多時,通常用switch語句替換if語句茶宵。

switch語句最簡單的形式就是把某個值與一個或若干個相同類型的值作比較:

switch some value to consider {
case value 1:
    respond to value 1
case value 2,
    value 3:
    respond to value 2 or 3
default:
    otherwise, do something else
}

switch語句都由多個 case 構(gòu)成危纫。為了匹配某些更特定的值,Swift 提供了幾種更復(fù)雜的匹配模式乌庶,這些模式將在本節(jié)的稍后部分提到种蝶。

每一個 case 都是代碼執(zhí)行的一條分支,這與if語句類似瞒大。與之不同的是螃征,switch語句會決定哪一條分支應(yīng)該被執(zhí)行。

switch語句必須是完備的透敌。這就是說盯滚,每一個可能的值都必須至少有一個 case 分支與之對應(yīng)。在某些不可能涵蓋所有值的情況下酗电,你可以使用默認(rèn)(default)分支滿足該要求魄藕,這個默認(rèn)分支必須在switch語句的最后面。

下面的例子使用switch語句來匹配一個名為someCharacter的小寫字符:

let someCharacter: Character = "e"
switch someCharacter {
case "a", "e", "i", "o", "u":
    print("\(someCharacter) is a vowel")
case "b", "c", "d", "f", "g", "h", "j", "k", "l", "m",
"n", "p", "q", "r", "s", "t", "v", "w", "x", "y", "z":
    print("\(someCharacter) is a consonant")
default:
    print("\(someCharacter) is not a vowel or a consonant")
}
// 輸出 "e is a vowel"

在這個例子中撵术,第一個 case 分支用于匹配五個元音背率,第二個 case 分支用于匹配所有的輔音。

由于為其它可能的字符寫 case 分支沒有實際的意義嫩与,因此在這個例子中使用了默認(rèn)分支來處理剩下的既不是元音也不是輔音的字符——這就保證了switch語句的完備性寝姿。

不存在隱式的貫穿(No Implicit Fallthrough)

與 C 語言和 Objective-C 中的switch語句不同,在 Swift 中划滋,當(dāng)匹配的 case 分支中的代碼執(zhí)行完畢后饵筑,程序會終止switch語句,而不會繼續(xù)執(zhí)行下一個 case 分支处坪。這也就是說根资,不需要在 case 分支中顯式地使用break語句。這使得switch語句更安全稻薇、更易用,也避免了因忘記寫break語句而產(chǎn)生的錯誤胶征。

注意: 雖然在Swift中break不是必須的塞椎,但你依然可以在 case 分支中的代碼執(zhí)行完畢前使用break跳出,詳情請參見Switch 語句中的 break睛低。
每一個 case 分支都必須包含至少一條語句案狠。像下面這樣書寫代碼是無效的服傍,因為第一個 case 分支是空的:

let anotherCharacter: Character = "a"
switch anotherCharacter {
case "a":
case "A":
    print("The letter A")
default:
    print("Not the letter A")
}
// this will report a compile-time error

不像 C 語言里的switch語句,在 Swift 中骂铁,switch語句不會同時匹配"a"和"A"吹零。相反的,上面的代碼會引起編譯期錯誤:case "a": does not contain any executable statements——這就避免了意外地從一個 case 分支貫穿到另外一個拉庵,使得代碼更安全灿椅、也更直觀。

一個 case 也可以包含多個模式钞支,用逗號把它們分開(如果太長了也可以分行寫):

switch some value to consider {
case value 1,
     value 2:
    statements
}

注意: 如果想要貫穿至特定的 case 分支中茫蛹,請使用fallthrough語句,詳情請參考貫穿(Fallthrough)烁挟。

區(qū)間匹配

case 分支的模式也可以是一個值的區(qū)間婴洼。下面的例子展示了如何使用區(qū)間匹配來輸出任意數(shù)字對應(yīng)的自然語言格式:

let approximateCount = 62
let countedThings = "moons orbiting Saturn"
var naturalCount: String
switch approximateCount {
case 0:
    naturalCount = "no"
case 1..<5:
    naturalCount = "a few"
case 5..<12:
    naturalCount = "several"
case 12..<100:
    naturalCount = "dozens of"
case 100..<1000:
    naturalCount = "hundreds of"
default:
    naturalCount = "many"
}
print("There are \(naturalCount) \(countedThings).")
// 輸出 "There are dozens of moons orbiting Saturn."

在上例中,approximateCount在一個switch聲明中被估值撼嗓。每一個case都與之進(jìn)行比較柬采。因為approximateCount落在了 12 到 100 的區(qū)間,所以naturalCount等于"dozens of"值且警,并且此后這段執(zhí)行跳出了switch聲明粉捻。

注意: 閉區(qū)間操作符(...)以及半開區(qū)間操作符(..<)功能被重載去返回IntervalType或Range。一個區(qū)間可以決定他是否包含特定的元素振湾,就像當(dāng)匹配一個switch聲明的case一樣杀迹。區(qū)間是一個連續(xù)值的集合,可以用for-in語句遍歷它押搪。

元組(Tuple)

我們可以使用元組在同一個switch語句中測試多個值树酪。元組中的元素可以是值,也可以是區(qū)間大州。另外续语,使用下劃線(_)來匹配所有可能的值。

下面的例子展示了如何使用一個(Int, Int)類型的元組來分類下圖中的點(x, y):

let somePoint = (1, 1)
switch somePoint {
case (0, 0):
    print("(0, 0) is at the origin")
case (_, 0):
    print("(\(somePoint.0), 0) is on the x-axis")
case (0, _):
    print("(0, \(somePoint.1)) is on the y-axis")
case (-2...2, -2...2):
    print("(\(somePoint.0), \(somePoint.1)) is inside the box")
default:
    print("(\(somePoint.0), \(somePoint.1)) is outside of the box")
}
// 輸出 "(1, 1) is inside the box"
coordinateGraphSimple.png

在上面的例子中厦画,switch語句會判斷某個點是否是原點(0, 0)疮茄,是否在紅色的x軸上,是否在黃色y軸上根暑,是否在一個以原點為中心的4x4的矩形里力试,或者在這個矩形外面。

不像 C 語言排嫌,Swift 允許多個 case 匹配同一個值畸裳。實際上,在這個例子中淳地,點(0, 0)可以匹配所有四個 case怖糊。但是帅容,如果存在多個匹配,那么只會執(zhí)行第一個被匹配到的 case 分支伍伤〔⑴牵考慮點(0, 0)會首先匹配case (0, 0),因此剩下的能夠匹配(0, 0)的 case 分支都會被忽視掉扰魂。

值綁定(Value Bindings)

case 分支的模式允許將匹配的值綁定到一個臨時的常量或變量麦乞,這些常量或變量在該 case 分支里就可以被引用了——這種行為被稱為值綁定(value binding)。

下面的例子展示了如何在一個(Int, Int)類型的元組中使用值綁定來分類下圖中的點(x, y):

let anotherPoint = (2, 0)
switch anotherPoint {
case (let x, 0):
    print("on the x-axis with an x value of \(x)")
case (0, let y):
    print("on the y-axis with a y value of \(y)")
case let (x, y):
    print("somewhere else at (\(x), \(y))")
}
// 輸出 "on the x-axis with an x value of 2"
coordinateGraphMedium.png

在上面的例子中阅爽,switch語句會判斷某個點是否在紅色的x軸上路幸,是否在黃色y軸上,或者不在坐標(biāo)軸上付翁。

這三個 case 都聲明了常量x和y的占位符简肴,用于臨時獲取元組anotherPoint的一個或兩個值。第一個 case ——case (let x, 0)將匹配一個縱坐標(biāo)為0的點百侧,并把這個點的橫坐標(biāo)賦給臨時的常量x砰识。類似的,第二個 case ——case (0, let y)將匹配一個橫坐標(biāo)為0的點佣渴,并把這個點的縱坐標(biāo)賦給臨時的常量y辫狼。

一旦聲明了這些臨時的常量,它們就可以在其對應(yīng)的 case 分支里引用辛润。在這個例子中膨处,它們用于簡化print(_:separator:terminator:)的書寫。

請注意砂竖,這個switch語句不包含默認(rèn)分支真椿。這是因為最后一個 case ——case let(x, y)聲明了一個可以匹配余下所有值的元組。這使得switch語句已經(jīng)完備了乎澄,因此不需要再書寫默認(rèn)分支突硝。

Where

case 分支的模式可以使用where語句來判斷額外的條件。

下面的例子把下圖中的點(x, y)進(jìn)行了分類:

let yetAnotherPoint = (1, -1)
switch yetAnotherPoint {
case let (x, y) where x == y:
    print("(\(x), \(y)) is on the line x == y")
case let (x, y) where x == -y:
    print("(\(x), \(y)) is on the line x == -y")
case let (x, y):
    print("(\(x), \(y)) is just some arbitrary point")
}
// 輸出 "(1, -1) is on the line x == -y"
coordinateGraphComplex.png

在上面的例子中置济,switch語句會判斷某個點是否在綠色的對角線x == y上解恰,是否在紫色的對角線x == -y上,或者不在對角線上浙于。

這三個 case 都聲明了常量x和y的占位符护盈,用于臨時獲取元組yetAnotherPoint的兩個值。這些常量被用作where語句的一部分羞酗,從而創(chuàng)建一個動態(tài)的過濾器(filter)腐宋。當(dāng)且僅當(dāng)where語句的條件為true時,匹配到的 case 分支才會被執(zhí)行。

就像是值綁定中的例子脏款,由于最后一個 case 分支匹配了余下所有可能的值,switch語句就已經(jīng)完備了裤园,因此不需要再書寫默認(rèn)分支撤师。

控制轉(zhuǎn)移語句(Control Transfer Statements)
控制轉(zhuǎn)移語句改變你代碼的執(zhí)行順序,通過它你可以實現(xiàn)代碼的跳轉(zhuǎn)拧揽。Swift 有五種控制轉(zhuǎn)移語句:

continue
break
fallthrough
return
throw
我們將會在下面討論continue剃盾、break和fallthrough語句。return語句將會在函數(shù)章節(jié)討論淤袜,throw語句會在錯誤拋出章節(jié)討論痒谴。

Continue

continue語句告訴一個循環(huán)體立刻停止本次循環(huán)迭代,重新開始下次循環(huán)迭代铡羡。就好像在說“本次循環(huán)迭代我已經(jīng)執(zhí)行完了”积蔚,但是并不會離開整個循環(huán)體。

注意: 在一個帶有條件和遞增的 for 循環(huán)體中烦周,調(diào)用continue語句后尽爆,迭代增量仍然會被計算求值。循環(huán)體繼續(xù)像往常一樣工作读慎,僅僅只是循環(huán)體中的執(zhí)行代碼會被跳過漱贱。
下面的例子把一個小寫字符串中的元音字母和空格字符移除,生成了一個含義模糊的短句:

let puzzleInput = "great minds think alike"
var puzzleOutput = ""
for character in puzzleInput.characters {
    switch character {
    case "a", "e", "i", "o", "u", " ":
        continue
    default:
        puzzleOutput.append(character)
    }
}
print(puzzleOutput)
// 輸出 "grtmndsthnklk"

在上面的代碼中夭委,只要匹配到元音字母或者空格字符幅狮,就調(diào)用continue語句,使本次循環(huán)迭代結(jié)束株灸,從新開始下次循環(huán)迭代崇摄。這種行為使switch匹配到元音字母和空格字符時不做處理,而不是讓每一個匹配到的字符都被打印蚂且。

Break

break語句會立刻結(jié)束整個控制流的執(zhí)行配猫。當(dāng)你想要更早的結(jié)束一個switch代碼塊或者一個循環(huán)體時,你都可以使用break語句杏死。

循環(huán)語句中的 break

當(dāng)在一個循環(huán)體中使用break時泵肄,會立刻中斷該循環(huán)體的執(zhí)行,然后跳轉(zhuǎn)到表示循環(huán)體結(jié)束的大括號(})后的第一行代碼淑翼。不會再有本次循環(huán)迭代的代碼被執(zhí)行腐巢,也不會再有下次的循環(huán)迭代產(chǎn)生。

Switch 語句中的 break

當(dāng)在一個switch代碼塊中使用break時玄括,會立即中斷該switch代碼塊的執(zhí)行冯丙,并且跳轉(zhuǎn)到表示switch代碼塊結(jié)束的大括號(})后的第一行代碼。

這種特性可以被用來匹配或者忽略一個或多個分支。因為 Swift 的switch需要包含所有的分支而且不允許有為空的分支胃惜,有時為了使你的意圖更明顯泞莉,需要特意匹配或者忽略某個分支。那么當(dāng)你想忽略某個分支時船殉,可以在該分支內(nèi)寫上break語句鲫趁。當(dāng)那個分支被匹配到時,分支內(nèi)的break語句立即結(jié)束switch代碼塊利虫。

注意: 當(dāng)一個switch分支僅僅包含注釋時挨厚,會被報編譯時錯誤。注釋不是代碼語句而且也不能讓switch分支達(dá)到被忽略的效果糠惫。你總是可以使用break來忽略某個分支疫剃。
下面的例子通過switch來判斷一個Character值是否代表下面四種語言之一。為了簡潔硼讽,多個值被包含在了同一個分支情況中巢价。

let numberSymbol: Character = "三"  // 簡體中文里的數(shù)字 3
var possibleIntegerValue: Int?
switch numberSymbol {
case "1", "?", "一", "?":
    possibleIntegerValue = 1
case "2", "?", "二", "?":
    possibleIntegerValue = 2
case "3", "?", "三", "?":
    possibleIntegerValue = 3
case "4", "?", "四", "?":
    possibleIntegerValue = 4
default:
    break
}
if let integerValue = possibleIntegerValue {
    print("The integer value of \(numberSymbol) is \(integerValue).")
} else {
    print("An integer value could not be found for \(numberSymbol).")
}
// 輸出 "The integer value of 三 is 3."

這個例子檢查numberSymbol是否是拉丁,阿拉伯固阁,中文或者泰語中的1到4之一蹄溉。如果被匹配到,該switch分支語句給Int?類型變量possibleIntegerValue設(shè)置一個整數(shù)值您炉。

當(dāng)switch代碼塊執(zhí)行完后柒爵,接下來的代碼通過使用可選綁定來判斷possibleIntegerValue是否曾經(jīng)被設(shè)置過值。因為是可選類型的緣故赚爵,possibleIntegerValue有一個隱式的初始值nil棉胀,所以僅僅當(dāng)possibleIntegerValue曾被switch代碼塊的前四個分支中的某個設(shè)置過一個值時,可選的綁定將會被判定為成功冀膝。

在上面的例子中唁奢,想要把Character所有的的可能性都枚舉出來是不現(xiàn)實的,所以使用default分支來包含所有上面沒有匹配到字符的情況窝剖。由于這個default分支不需要執(zhí)行任何動作麻掸,所以它只寫了一條break語句。一旦落入到default分支中后赐纱,break語句就完成了該分支的所有代碼操作脊奋,代碼繼續(xù)向下,開始執(zhí)行if let語句疙描。

貫穿(Fallthrough)

Swift 中的switch不會從上一個 case 分支落入到下一個 case 分支中诚隙。相反,只要第一個匹配到的 case 分支完成了它需要執(zhí)行的語句起胰,整個switch代碼塊完成了它的執(zhí)行久又。相比之下,C 語言要求你顯式地插入break語句到每個switch分支的末尾來阻止自動落入到下一個 case 分支中。Swift 的這種避免默認(rèn)落入到下一個分支中的特性意味著它的switch 功能要比 C 語言的更加清晰和可預(yù)測地消,可以避免無意識地執(zhí)行多個 case 分支從而引發(fā)的錯誤炉峰。

如果你確實需要 C 風(fēng)格的貫穿的特性,你可以在每個需要該特性的 case 分支中使用fallthrough關(guān)鍵字脉执。下面的例子使用fallthrough來創(chuàng)建一個數(shù)字的描述語句讲冠。

let integerToDescribe = 5
var description = "The number \(integerToDescribe) is"
switch integerToDescribe {
case 2, 3, 5, 7, 11, 13, 17, 19:
    description += " a prime number, and also"
    fallthrough
default:
    description += " an integer."
}
print(description)
// 輸出 "The number 5 is a prime number, and also an integer."

這個例子定義了一個String類型的變量description并且給它設(shè)置了一個初始值。函數(shù)使用switch邏輯來判斷integerToDescribe變量的值适瓦。當(dāng)integerToDescribe的值屬于列表中的質(zhì)數(shù)之一時,該函數(shù)添加一段文字在description后谱仪,來表明這個是數(shù)字是一個質(zhì)數(shù)玻熙。然后它使用fallthrough關(guān)鍵字來“貫穿”到default分支中。default分支添加一段額外的文字在description的最后疯攒,至此switch代碼塊執(zhí)行完了嗦随。

如果integerToDescribe的值不屬于列表中的任何質(zhì)數(shù),那么它不會匹配到第一個switch分支敬尺。而這里沒有其他特別的分支情況枚尼,所以integerToDescribe匹配到包含所有的default分支中。

當(dāng)switch代碼塊執(zhí)行完后砂吞,使用print(_:separator:terminator:)函數(shù)打印該數(shù)字的描述署恍。在這個例子中,數(shù)字5被準(zhǔn)確的識別為了一個質(zhì)數(shù)蜻直。

注意: fallthrough關(guān)鍵字不會檢查它下一個將會落入執(zhí)行的 case 中的匹配條件盯质。fallthrough簡單地使代碼執(zhí)行繼續(xù)連接到下一個 case 中的執(zhí)行代碼,這和 C 語言標(biāo)準(zhǔn)中的switch語句特性是一樣的概而。

帶標(biāo)簽的語句

在 Swift 中呼巷,你可以在循環(huán)體和switch代碼塊中嵌套循環(huán)體和switch代碼塊來創(chuàng)造復(fù)雜的控制流結(jié)構(gòu)。然而赎瑰,循環(huán)體和switch代碼塊兩者都可以使用break語句來提前結(jié)束整個方法體王悍。因此,顯式地指明break語句想要終止的是哪個循環(huán)體或者switch代碼塊餐曼,會很有用压储。類似地,如果你有許多嵌套的循環(huán)體源譬,顯式指明continue語句想要影響哪一個循環(huán)體也會非常有用渠脉。

為了實現(xiàn)這個目的,你可以使用標(biāo)簽來標(biāo)記一個循環(huán)體或者switch代碼塊瓶佳,當(dāng)使用break或者continue時芋膘,帶上這個標(biāo)簽,可以控制該標(biāo)簽代表對象的中斷或者執(zhí)行。

產(chǎn)生一個帶標(biāo)簽的語句是通過在該語句的關(guān)鍵詞的同一行前面放置一個標(biāo)簽为朋,并且該標(biāo)簽后面還需帶著一個冒號臂拓。下面是一個while循環(huán)體的語法,同樣的規(guī)則適用于所有的循環(huán)體和switch代碼塊习寸。

label name: while condition { statements }
下面的例子是在一個帶有標(biāo)簽的while循環(huán)體中調(diào)用break和continue語句胶惰,該循環(huán)體是前面章節(jié)中蛇和梯子的改編版本。這次霞溪,游戲增加了一條額外的規(guī)則:

為了獲勝孵滞,你必須剛好落在第 25 個方塊中。
如果某次擲骰子使你的移動超出第 25 個方塊鸯匹,你必須重新擲骰子坊饶,直到你擲出的骰子數(shù)剛好使你能落在第 25 個方塊中。

游戲的棋盤和之前一樣:

snakesAndLadders.png

finalSquare殴蓬、board匿级、square和diceRoll值被和之前一樣的方式初始化:

let finalSquare = 25
var board = [Int](count: finalSquare + 1, repeatedValue: 0)
board[03] = +08; board[06] = +11; board[09] = +09; board[10] = +02
board[14] = -10; board[19] = -11; board[22] = -02; board[24] = -08
var square = 0
var diceRoll = 0

這個版本的游戲使用while循環(huán)體和switch方法塊來實現(xiàn)游戲的邏輯。while循環(huán)體有一個標(biāo)簽名gameLoop染厅,來表明它是蛇與梯子的主循環(huán)痘绎。

該while循環(huán)體的條件判斷語句是while square !=finalSquare,這表明你必須剛好落在方格25中肖粮。

gameLoop: while square != finalSquare {
    if ++diceRoll == 7 { diceRoll = 1 }
    switch square + diceRoll {
    case finalSquare:
        // 到達(dá)最后一個方塊孤页,游戲結(jié)束
        break gameLoop
    case let newSquare where newSquare > finalSquare:
        // 超出最后一個方塊,再擲一次骰子
        continue gameLoop
    default:
        // 本次移動有效
        square += diceRoll
        square += board[square]
    }
}
print("Game over!")

每次循環(huán)迭代開始時擲骰子涩馆。與之前玩家擲完骰子就立即移動不同散庶,這里使用了switch來考慮每次移動可能產(chǎn)生的結(jié)果,從而決定玩家本次是否能夠移動凌净。

如果骰子數(shù)剛好使玩家移動到最終的方格里悲龟,游戲結(jié)束。break gameLoop語句跳轉(zhuǎn)控制去執(zhí)行while循環(huán)體后的第一行代碼冰寻,游戲結(jié)束须教。
如果骰子數(shù)將會使玩家的移動超出最后的方格,那么這種移動是不合法的斩芭,玩家需要重新擲骰子轻腺。continue gameLoop語句結(jié)束本次while循環(huán)的迭代,開始下一次循環(huán)迭代划乖。
在剩余的所有情況中贬养,骰子數(shù)產(chǎn)生的都是合法的移動。玩家向前移動骰子數(shù)個方格琴庵,然后游戲邏輯再處理玩家當(dāng)前是否處于蛇頭或者梯子的底部误算。本次循環(huán)迭代結(jié)束仰美,控制跳轉(zhuǎn)到while循環(huán)體的條件判斷語句處,再決定是否能夠繼續(xù)執(zhí)行下次循環(huán)迭代儿礼。
注意: 如果上述的break語句沒有使用gameLoop標(biāo)簽咖杂,那么它將會中斷switch代碼塊而不是while循環(huán)體。使用gameLoop標(biāo)簽清晰的表明了break想要中斷的是哪個代碼塊蚊夫。 同時請注意诉字,當(dāng)調(diào)用continue gameLoop去跳轉(zhuǎn)到下一次循環(huán)迭代時,這里使用gameLoop標(biāo)簽并不是嚴(yán)格必須的知纷。因為在這個游戲中壤圃,只有一個循環(huán)體,所以continue語句會影響到哪個循環(huán)體是沒有歧義的琅轧。然而伍绳,continue語句使用gameLoop標(biāo)簽也是沒有危害的。這樣做符合標(biāo)簽的使用規(guī)則鹰晨,同時參照旁邊的break gameLoop,能夠使游戲的邏輯更加清晰和易于理解止毕。

提前退出
像if語句一樣模蜡,guard的執(zhí)行取決于一個表達(dá)式的布爾值。我們可以使用guard語句來要求條件必須為真時扁凛,以執(zhí)行g(shù)uard語句后的代碼忍疾。不同于if語句,一個guard語句總是有一個else分句谨朝,如果條件不為真則執(zhí)行else分句中的代碼卤妒。

func greet(person: [String: String]) {
    guard let name = person["name"] else {
        return
    }
    print("Hello \(name)")

    guard let location = person["location"] else {
        print("I hope the weather is nice near you.")
        return
    }
    print("I hope the weather is nice in \(location).")
}
greet(["name": "John"])
// 輸出 "Hello John!"
// 輸出 "I hope the weather is nice near you."
greet(["name": "Jane", "location": "Cupertino"])
// 輸出 "Hello Jane!"
// 輸出 "I hope the weather is nice in Cupertino."

如果guard語句的條件被滿足,則在保護語句的封閉大括號結(jié)束后繼續(xù)執(zhí)行代碼字币。任何使用了可選綁定作為條件的一部分并被分配了值的變量或常量對于剩下的保護語句出現(xiàn)的代碼段是可用的则披。

如果條件不被滿足,在else分支上的代碼就會被執(zhí)行洗出。這個分支必須轉(zhuǎn)移控制以退出guard語句出現(xiàn)的代碼段士复。它可以用控制轉(zhuǎn)移語句如return,break,continue或者throw做這件事,或者調(diào)用一個不返回的方法或函數(shù)翩活,例如fatalError()阱洪。

相比于可以實現(xiàn)同樣功能的if語句,按需使用guard語句會提升我們代碼的可靠性菠镇。它可以使你的代碼連貫的被執(zhí)行而不需要將它包在else塊中冗荸,它可以使你處理違反要求的代碼接近要求。

檢測 API 可用性
Swift 有檢查 API 可用性的內(nèi)置支持利耍,這可以確保我們不會不小心地使用對于當(dāng)前部署目標(biāo)不可用的 API蚌本。

編譯器使用 SDK 中的可用信息來驗證我們的代碼中使用的所有 API 在項目指定的部署目標(biāo)上是否可用盔粹。如果我們嘗試使用一個不可用的 API,Swift 會在編譯時報錯魂毁。

我們使用一個可用性條件在一個if或guard語句中去有條件的執(zhí)行一段代碼玻佩,這取決于我們想要使用的 API 是否在運行時是可用的。編譯器使用從可用性條件語句中獲取的信息去驗證在代碼塊中調(diào)用的 API 是否都可用席楚。

if #available(iOS 9, OSX 10.10, *) {
    // 在 iOS 使用 iOS 9 的 API, 在 OS X 使用 OS X v10.10 的 API
} else {
    // 使用先前版本的 iOS 和 OS X 的 API
}

以上可用性條件指定在 iOS咬崔,if段的代碼僅僅在 iOS 9 及更高可運行;在 OS X烦秩,僅在 OS X v10.10 及更高可運行垮斯。最后一個參數(shù),*只祠,是必須的并且指定在任何其他平臺上兜蠕,if段的代碼在最小可用部署目標(biāo)指定項目中執(zhí)行。

在它普遍的形式中抛寝,可用性條件獲取了平臺名字和版本的清單熊杨。平臺名字可以是iOS,OSX或watchOS盗舰。除了特定的主板本號像 iOS 8晶府,我們可以指定較小的版本號像iOS 8.3 以及 OS X v10.10.3。

if #available(`platform name` `version`, `...`, *) {
    `statements to execute if the APIs are available`
} else {
    `fallback statements to execute if the APIs are unavailable`
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末钻趋,一起剝皮案震驚了整個濱河市川陆,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌蛮位,老刑警劉巖较沪,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異失仁,居然都是意外死亡尸曼,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門萄焦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來骡苞,“玉大人,你說我怎么就攤上這事楷扬〗庥模” “怎么了?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵烘苹,是天一觀的道長躲株。 經(jīng)常有香客問我,道長镣衡,這世上最難降的妖魔是什么霜定? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任档悠,我火速辦了婚禮,結(jié)果婚禮上望浩,老公的妹妹穿的比我還像新娘辖所。我一直安慰自己,他們只是感情好磨德,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布缘回。 她就那樣靜靜地躺著,像睡著了一般典挑。 火紅的嫁衣襯著肌膚如雪酥宴。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天您觉,我揣著相機與錄音拙寡,去河邊找鬼。 笑死琳水,一個胖子當(dāng)著我的面吹牛肆糕,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播在孝,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼诚啃,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了浑玛?” 一聲冷哼從身側(cè)響起绍申,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤噩咪,失蹤者是張志新(化名)和其女友劉穎顾彰,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體胃碾,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡涨享,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了仆百。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片厕隧。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖俄周,靈堂內(nèi)的尸體忽然破棺而出吁讨,到底是詐尸還是另有隱情,我是刑警寧澤峦朗,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布建丧,位于F島的核電站,受9級特大地震影響波势,放射性物質(zhì)發(fā)生泄漏翎朱。R本人自食惡果不足惜橄维,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望拴曲。 院中可真熱鬧争舞,春花似錦、人聲如沸澈灼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽蕉汪。三九已至流译,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間者疤,已是汗流浹背福澡。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留驹马,地道東北人革砸。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像糯累,于是被迫代替她去往敵國和親算利。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容