Swift5.x-控制流(中文文档)

引言

继续学习Swift文档,从上一章节:集合类型,我们学习了Swift集合相关的内容。现在,我们学习Swift的控制流相关的内容。由于篇幅较长,这里分篇来记录,接下来,开始吧!

如果你已经熟悉并掌握了条件控制相关内容,请参阅下一章节:函数

控制流

Swift提供了多种控制流语句。这些包括while循环来多次执行一个任务;if、guard和switch语句根据特定条件执行不同的代码分支;以及诸如break和continue这样的语句将执行流转移到代码中的另一个点。

Swift还提供了一个for-in循环,可以方便地遍历数组、字典、范围、字符串和其他序列。

Swift的switch声明要比许多类似c语言的版本强大得多。用例可以匹配许多不同的模式,包括间隔匹配、元组和特定类型的强制类型转换。switch案例中的匹配值可以绑定到临时常量或变量,以便在案例主体中使用,复杂的匹配条件可以用where子句表示。

1 For-in循环

可以使用for-in循环迭代序列,例如数组中的项、数字范围或字符串中的字符。

这个例子使用for-in循环来遍历数组中的项:

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

您还可以遍历字典来访问它的键-值对。在迭代字典时,字典中的每个项都作为(键、值)元组返回,您可以将(键、值)元组的成员分解为显式命名的常量,以便在for-in循环体中使用。在下面的代码示例中,字典的键分解为一个名为animalName的常量,字典的值分解为一个名为legCount的常量。

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

字典的内容本质上是无序的,对它们进行迭代不能保证检索它们的顺序。特别是,向字典中插入条目的顺序并不定义迭代它们的顺序。有关数组和字典的更多信息,请参见集合类型

您还可以使用具有数值范围的for-in循环。这个示例打印一个5次表中的前几项:

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

被迭代的序列是一个从1到5(包括5)的数字范围,如使用闭范围操作符(…)所示。index的值设置为范围(1)中的第一个数字,然后执行循环中的语句。在本例中,循环只包含一条语句,它从索引的当前值的五次表中打印一个条目。在执行语句后,index的值被更新为包含范围(2)中的第二个值,并且再次调用print(_:separator:terminator:)函数。这个过程一直持续到范围结束。

在上面的示例中,index是一个常量,它的值在循环的每次迭代开始时自动设置。因此,索引在使用之前不需要声明。只需在循环声明中包含它,就可以隐式地声明它,不需要let声明关键字。

如果不需要序列中的每个值,可以使用下划线代替变量名来忽略这些值。

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

上面的示例计算了一个数的几次方(在本例中是3的10次方)。它将起始值1(即3的0次方)乘以3,10次,使用从1开始到10结束的封闭范围。对于这种计算,每次循环中单独的计数器值都是不必要的—代码只执行正确次数的循环。用来代替循环变量的下划线字符(_)会导致单个值被忽略,并且不会在循环的每次迭代中提供对当前值的访问。

在某些情况下,您可能不想使用包括两个端点的封闭范围。考虑在表面上画每分钟的刻度。您需要画60个刻度,从0分钟开始。使用半开范围操作符(..<)来包含下界,而不是上界。有关范围的更多信息,请参见范围运算符

let minutes = 60
for tickMark in 0..<minutes {
    // render the tick mark each minute (60 times)
}

有些用户可能想在他们的UI中减少标记。他们可以选择每5分钟打一个分数。使用stride(from:to:by:)函数跳过不需要的标记。

let minuteInterval = 5
for tickMark in stride(from: 0, to: minutes, by: minuteInterval) {
    // render the tick mark every 5 minutes (0, 5, 10, 15 ... 45, 50, 55)
}

通过使用stride(from:through:by:)也可以使用封闭范围:

let hours = 12
let hourInterval = 3
for tickMark in stride(from: 3, through: hours, by: hourInterval) {
    // render the tick mark every 3 hours (3, 6, 9, 12)
}

2 While循环

while循环执行一组语句,直到条件变为false为止。当在第一次迭代开始之前不知道迭代的数量时,最好使用这种类型的循环。Swift提供了两种while循环:

  • while 在每次遍历循环开始时计算其条件。
  • repeat-while——在每次循环结束时计算其条件。

2.1 While

while循环从计算单个条件开始。如果条件为真,则重复一组语句,直到条件为假。

下面是while循环的一般形式:

while condition {
    statements
}

这个例子玩了一个简单的蛇和梯zi(掘金里面被定义为违规内容,所以用拼音代替)的游戏(也称为滑道和梯zi):


image.png

游戏规则如下:

  • 棋盘有25个方格,目标是落在或超过第25个方格。
  • 玩家的起始方块是“square zero”,它位于棋盘左下角。
  • 每一回合,你掷一个六面骰子,并移动该数目的方块,遵循水平路径指示,由上面的虚线箭头。
  • 如果你的回合结束在梯zi的底部,你就向上爬。
  • 如果你的回合结束在一条蛇的头部,你移动下那条蛇。

游戏板由Int值数组表示。它的大小基于一个名为finalSquare的常量,该常量用于初始化数组,并在本例中稍后检查win条件。因为玩家从棋盘开始,在“square zero”上,棋盘被初始化为26个0 Int值,而不是25。

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

一些方块会为蛇和梯zi设置更具体的值。有梯zi底座的方块有一个正数可以让你在棋盘上向上移动,而蛇头的方块有一个负数可以让你在棋盘上向下移动。

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包含梯zi的底部,它可以把你移动到正方形11。为了表示这个,board[03]等于+08,这等于一个整数8(3和11之间的差)。为了对齐值和语句,一元加号操作符(+i)显式地与一元减号操作符(-i)一起使用,小于10的数字用零填充。(这两种文体技巧都不是绝对必要的,但它们能使代码更整洁。)

var square = 0
var diceRoll = 0
while square < finalSquare {
    // roll the dice
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    // move by the rolled amount
    square += diceRoll
    if square < board.count {
        // if we're still on the board, move up or down for a snake or a ladder
        square += board[square]
    }
}
print("Game over!")

上面的示例使用了一种非常简单的掷骰子方法。它不是生成随机数,而是从diceRoll值0开始。每次在while循环中,diceRoll都会增加1,然后检查它是否变得太大了。每当这个返回值等于7时,掷骰就会变得太大,并被重置为1。结果是一个diceRoll值序列,总是1、2、3、4、5、6、1、2等等。

滚动骰子后,玩家前进的diceRoll广场。有可能是掷骰子把玩家移动到了25方,这样游戏就结束了。为了处理这种情况,代码检查正方形小于board数组的count属性。如果方块是有效的,存储在棋盘[方块]中的值将被添加到当前方块值中,以使玩家在任何梯zi或蛇上上下移动。

注意
如果这个检查没有执行,board[square]可能会尝试访问一个超出board数组边界的值,这将触发一个运行时错误。

此时,当前while循环执行结束,并检查循环的条件,以确定是否应该再次执行该循环。如果玩家已经移动到或超过了平方数25,循环的条件将计算为false,游戏结束。

在这种情况下使用while循环是合适的,因为游戏的长度在while循环开始时并不清楚。相反,循环将一直执行到满足特定条件为止。

2.2 Repeat-While

while循环的另一个变体,称为repeat-while循环,在考虑循环条件之前,会先执行一次循环循环。 然后,它将继续repeat-while循环,直到条件为假。

注意
Swift中的repeat-while循环类似于其他语言中的do-while循环。

这是repeat-while循环的一般形式:

repeat {
    statements
} while condition

这又是“蛇和梯zi”示例,写为repeat-while循环而不是while循环。 finalSquare,board,square和diceRoll的值的初始化方式与while循环完全相同。

let finalSquare = 25
var board = [Int](repeating: 0, count: finalSquare + 1)
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

在此版本的游戏中,循环中的第一个动作是检查梯zi或蛇。 棋盘上没有梯zi可使玩家直奔25格,因此不可能通过爬梯zi来赢得比赛。 因此,安全地检查蛇或梯zi是循环中的第一步。

在游戏开始时,玩家处于“零平方”。 board [0]始终等于0,并且无效。

repeat {
    // move up or down for a snake or ladder
    square += board[square]
    // roll the dice
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    // move by the rolled amount
    square += diceRoll
} while square < finalSquare
print("Game over!")

代码检查完蛇和梯zi后,掷骰子,并由diceRoll方块使玩家向前移动。 然后,当前循环执行结束。

循环的条件(square<finalSquare时)与以前相同,但是这次直到循环的第一次运行结束时才进行评估。 与之前示例中的while循环相比,repeat-while循环的结构更适合此游戏。 在上面的repeat-while循环中,square+ = board [square]总是在循环的while条件确认正方形仍在板上后立即执行。 此行为消除了对前面所述游戏的while循环版本中看到的数组边界检查的需要。

3 条件语句

根据某些条件执行不同的代码段通常很有用。 您可能想在发生错误时运行额外的代码,或者在值变得太大或太低时显示一条消息。 为此,您需要将部分代码作为条件。

Swift提供了两种向代码中添加条件分支的方法:if语句和switch语句。 通常,您使用if语句评估只有少数可能结果的简单条件。 switch语句更适合于具有多个可能排列的更复杂条件,并且在模式匹配可以帮助选择合适的代码分支来执行的情况下很有用。

3.1 If

以最简单的形式,if语句具有单个if条件。 仅当条件为真时,它才执行一组语句。

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

上面的示例检查温度是否小于或等于32华氏度(水的冰点)。 如果是,则会打印一条消息。 否则,不会打印任何消息,并且if语句的右大括号后代码将继续执行。

如果if条件为false,则if语句可以提供另一组语句,称为else子句。 这些语句由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.")
}
// Prints "It's not that cold. Wear a t-shirt."

这两个分支之一始终被执行。 由于温度已升高到40华氏度,因此不再足够寒冷以致建议戴围巾,因此触发了else分支。

您可以将多个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.")
}
// Prints "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.")
}

因为温度既不太冷也不太热,无法触发if or else if条件,所以不会打印任何消息。

3.2 Switch

switch语句考虑一个值,并将其与几种可能的匹配模式进行比较。 然后,它根据成功匹配的第一个模式执行适当的代码块。 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关键字开头。 除了与特定值进行比较之外,Swift还为每种情况提供了几种方法来指定更复杂的匹配模式。 这些选项将在本章后面介绍。

就像if语句的主体一样,每种情况都是代码执行的单独分支。 switch语句确定应选择哪个分支。 此过程打开要考虑的值。

每个switch语句必须详尽。 即,所考虑的类型的每个可能的值都必须与切换条件之一匹配。 如果不适合为每个可能的值都提供一个case,则可以定义一个默认case,以涵盖未明确寻址的所有值。 该默认情况由default关键字指示,并且必须始终显示在最后。

此示例使用switch语句考虑一个称为someCharacter的小写字符:

let someCharacter: Character = "z"
switch someCharacter {
case "a":
    print("The first letter of the alphabet")
case "z":
    print("The last letter of the alphabet")
default:
    print("Some other character")
}
// Prints "The last letter of the alphabet"

switch语句的第一个字母与英文字母的第一个字母a匹配,第二个字母与最后一个字母z匹配。 因为switch必须为每个可能的字符而不是每个字母字符都具有case,所以该switch语句使用默认的case来匹配除a和z之外的所有字符。 此规定确保switch语句是详尽无遗的。

3.3 无需隐式的Fallthrough

与C和Objective-C中的switch语句相比,Swift中的switch语句不会执行到每种情况的底部,默认情况下不会执行下一种情况。 相反,整个switch语句将在第一个匹配的switch情况完成后立即完成其执行,而无需显式的break语句。 这使得switch语句比C语言中的语句更安全和易于使用,并且避免了错误执行多个switch情况。

注意
尽管在Swift中不需要break,但是您可以使用break语句来匹配和忽略特定的情况,或者在该情况完成执行之前从匹配的情况中中断。 有关详细信息,请参见Break in a Switch Statement

每个案例的主体必须至少包含一个可执行语句。 编写以下代码是无效的,因为第一种情况为空:

let anotherCharacter: Character = "a"
switch anotherCharacter {
case "a": // Invalid, the case has an empty body
case "A":
    print("The letter A")
default:
    print("Not the letter A")
}
// This will report a compile-time error.

注意
OC可以这么写,不会编译报错。

与C中的switch语句不同,此switch语句不同时匹配“ a”和“ A”。 而是,报告一个编译时错误,情况为“ a”:不包含任何可执行语句。 这种方法避免了从一个案例到另一个案例的意外失败,并使意图更清晰,代码更安全成为可能。

要使用同时匹配“ a”和“ A”的单个case进行切换,请将两个值组合为复合case,并用逗号分隔这些值。

let anotherCharacter: Character = "a"
switch anotherCharacter {
case "a", "A":
    print("The letter A")
default:
    print("Not the letter A")
}
// Prints "The letter A"

为了便于阅读,还可以将多行案例写在多行上。 有关复合案例的更多信息,请参见复合案例

注意
要在特定切换条件的末尾显式的Fallthrough,请使用fallthrough关键字,如Fallthrough中所述。

3.4 间隔匹配

可以检查切换案例中的值是否包含在间隔中。 本示例使用数字间隔为任意大小的数字提供自然语言计数:

let approximateCount = 62
let countedThings = "moons orbiting Saturn"
let 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).")
// Prints "There are dozens of moons orbiting Saturn."

在上面的示例中,在switch语句中计算了近似计数。 每种情况都会将该值与一个数字或间隔进行比较。 由于近似计数的值介于12到100之间,因此为naturalCount分配了"dozens of"值,并且将执行转移到switch语句之外。

3.5 元组

您可以使用元组在同一switch语句中测试多个值。 可以针对不同的值或值的间隔测试元组的每个元素。 或者,使用下划线字符(_)(也称为通配符模式)来匹配任何可能的值。

下面的示例采用一个(x,y)点,表示为类型为(Int,Int)的简单元组,并将其分类在该示例之后的图形上。

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

switch语句确定点是在原点(0,0),在红色x轴上,在橙色y轴上,在以原点为中心的蓝色4 x 4框内还是在框外 。

与C不同,Swift允许多个switch cases判断相同的一个或多个值。 实际上,在此示例中,点(0,0)可以匹配所有四种情况。 但是,如果可能有多个匹配项,则始终使用第一个匹配情况。 点(0,0)首先匹配case(0,0),因此所有其他匹配cases都将被忽略。

3.6 值绑定

switch case可以命名临时常量或变量匹配的一个或多个值,以用于case的主体中。 这种行为称为值绑定,因为值绑定到case主体内的临时常量或变量。

下面的示例获取一个(x,y)点,表示为类型(Int,Int)的元组,并将其归类到下面的图形上:

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))")
}
// Prints "on the x-axis with an x value of 2"
image

switch语句确定该点是在红色的x轴上,在橙色的y轴上还是在其他位置(均不在两个轴上)。

这三个switch case声明了占位符常量x和y,它们暂时采用来自anotherPoint的一个或两个元组值。 第一种情况为case(let x,0),它匹配y值为0的任何点,并将该点的x值分配给临时常数x。 同样,第二种情况,即情况(0,设y),匹配x值为0的任何点,并将该点的y值分配给临时常数y。

声明临时常量后,可以在案例的代码块中使用它们。 在这里,它们用于打印点的分类。

该switch语句没有默认情况。 最后一种情况,即let(x,y),声明了两个可以匹配任何值的占位符常量的元组。 因为anotherPoint始终是两个值的元组,所以这种情况与所有可能的剩余值匹配,并且不需要缺省情况来使switch语句穷举。

3.7 Where

switch case可以使用where子句检查其他条件。

下面的示例在下图上对(x,y)点进行了分类:

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")
}
// Prints "(1, -1) is on the line x == -y"
image

switch语句确定该点是在x == y的绿色对角线上,在x == -y的紫色对角线上还是在这两个点上。

这三个switch case声明了占位符常量x和y,它们分别采用了yetAnotherPoint中的两个元组值。 这些常量用作where子句的一部分,以创建动态过滤器。 仅当where子句的条件评估为该值时,switch case才匹配point的当前值。

与前面的示例一样,最终情况匹配所有可能的剩余值,因此不需要默认情况来使switch语句穷举。

3.8 复合Cases

共享同一主体的多个switch case可通过在case后写入多个模式(每个模式之间使用逗号)来组合。 如果任何模式都匹配,则认为case匹配。 如果列表很长,可以将模式写在多行上。 例如:

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")
}
// Prints "e is a vowel"

switch语句的首字母大写与英语中的所有五个小写元音匹配。 同样,它的第二种情况匹配所有小写的英语辅音。 最后,默认大小写匹配任何其他字符。

复合案例还可以包括值绑定。 复合案例的所有模式都必须包含相同的值绑定集,并且每个绑定必须从复合案例的所有模式中获取相同类型的值。 这样可以确保,无论复合案例的哪一部分匹配,案例主体中的代码始终可以访问绑定的值,并且该值始终具有相同的类型。

let stillAnotherPoint = (9, 0)
switch stillAnotherPoint {
case (let distance, 0), (0, let distance):
    print("On an axis, \(distance) from the origin")
default:
    print("Not on an axis")
}
// Prints "On an axis, 9 from the origin"

上面的情况有两种模式:(let distance,0)匹配x轴上的点,(0,let distance)匹配y轴上的点。 两种模式都包含distance的绑定,并且两种模式中的distance都是整数,这意味着案例主体中的代码始终可以访问distance值。

4 控制转移语句

通过将控制权从一个代码段转移到另一个代码段,控制转移语句改变了代码执行的顺序。 Swift有五个控制转移语句:

continue
break
fallthrough
return
throw

continue、break和fallthrough语句描述如下。return语句的描述在 Functions里,throw语句可以参见Propagating Errors Using Throwing Functions

4.1 Continue

Continue语句告诉循环停止正在执行的操作,并在循环的下一次迭代开始时重新开始。 它说“我完成了当前的循环迭代”,而没有完全离开循环。

下面的示例从小写字符串中删除所有元音和空格,以创建一个神秘的难题短语:

let puzzleInput = "great minds think alike"
var puzzleOutput = ""
let charactersToRemove: [Character] = ["a", "e", "i", "o", "u", " "]
for character in puzzleInput {
    if charactersToRemove.contains(character) {
        continue
    }
    puzzleOutput.append(character)
}
print(puzzleOutput)
// Prints "grtmndsthnklk"

上面的代码在与元音或空格匹配时会调用continue关键字,从而导致循环的当前迭代立即结束并直接跳至下一个迭代的开始。

4.2 Break

break语句立即结束整个控制流语句的执行。 当您要比其他情况更早地终止switch或loop语句的执行时,可以在switch或loop语句内使用break语句。

4.2.1 循环语句里的Break

在循环语句中使用break时,会立即结束循环的执行,并在循环的右括号(})之后将控制权转移给代码。 不会执行循环的当前迭代中的其他代码,也不会启动循环的其他迭代。

4.2.2 Switch语句里的Break

在switch语句内使用break时,会导致switch语句立即结束其执行,并将控制权转移到switch语句的右括号(})之后。

此行为可用于匹配和忽略switch语句中的一个或多个情况。 由于Swift的switch语句是详尽无遗的,并且不允许空白的案例,因此有时有必要故意匹配并忽略案例,以使意图明确。 您可以通过将break语句作为要忽略的案例的整体来编写。 当该案例与switch语句匹配时,案例内的break语句将立即终止switch语句的执行。

注意
仅包含注释的switch case将报告为编译时错误。 注释不是语句,不会导致switch case被忽略。 始终使用break语句忽略switch case。

下面的示例打开一个Character值,并确定它是否代表四种语言之一的数字符号。 为简便起见,在一个switch case中包含多个值。

let numberSymbol: Character = "三"  // Chinese symbol for the number 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).")
}
// Prints "The integer value of 三 is 3."

本示例检查numberSymbol以确定数字1到4是拉丁,阿拉伯,中文还是泰文符号。如果找到匹配项,则switch语句的一种情况将设置可选的Int? 变量,可能为一个可选的整数值。

在switch语句完成执行之后,该示例使用可选绑定来确定是否找到了值。 PossibleIntegerValue变量是可选类型,因此具有隐式初始值nil,因此,只有在switch语句的前四种情况之一将可能性IntegerValue设置为实际值时,可选绑定才会成功。

由于在上面的示例中列出所有可能的Character值不切实际,因此默认情况下会处理所有不匹配的字符。 此默认情况不需要执行任何操作,因此它以单个break语句为主体编写。 匹配默认大小写后,break语句将结束switch语句的执行,并且代码将从if let语句继续执行。

4.3 Fallthrough

在Swift中,switch语句不会落入每种情况的底部,也不会落入下一种情况。 即,第一个匹配的情况一旦完成,整个switch语句就完成其执行。 相比之下,C要求您在每个switch情况的末尾插入一个显式的break语句,以防止失败。 避免默认的失败,意味着Swift的switch语句比C语言中的对应语句更加简洁和可预测,因此避免了错误地执行多个switch情况。

如果您需要C风格的fallthrough行为,则可以使用fallthrough关键字视情况选择加入此行为。 下面的示例使用fallthrough来创建数字的文本描述。

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)
// Prints "The number 5 is a prime number, and also an integer."

本示例声明一个名为description的新String变量,并为其分配一个初始值。 然后,该函数使用switch语句判断integerToDescribe的值。 如果integerToDescribe的值是列表中的质数之一,则该函数会将文本追加到描述末尾,以指出该数字是质数。 然后,它也使用fallthrough关键字来“落入”默认大小写。 默认情况下,在描述末尾添加了一些额外的文本,并且switch语句已完成。

除非integerToDescribe的值在已知质数列表中,否则第一个switch case根本不匹配。 因为没有其他特定情况,所以integerToDescribe与默认情况匹配。
在switch语句执行完之后,使用print(_:separator:terminator :)函数打印号码的描述。 在此示例中,数字5被正确标识为质数。

注意
fallthrough关键字不检查导致执行陷入的switch case的条件。 fallthrough关键字仅导致代码执行直接移至下一个case(或默认case)块内的语句,就像C的标准switch语句行为一样。

4.4 Labeled语句

在Swift中,您可以将循环和条件语句嵌套在其他循环和条件语句中,以创建复杂的控制流结构。 但是,循环和条件语句都可以使用break语句提前结束执行。 因此,有时对于明确要让break语句终止的循环或条件语句很有用。 同样,如果您有多个嵌套循环,则明确说明continue语句应该影响哪个循环会很有用。

为了实现这些目标,可以使用语句label标记循环语句或条件语句。 对于条件语句,可以将语句label与break语句一起使用以结束带label的语句的执行。 对于循环语句,可以将语句label与break或continue语句一起使用,以结束或继续执行带label的语句。

带label的语句是通过在与该语句的介绍者关键字相同的行上放置一个标签来表示的,后跟一个冒号。 这是一个while循环的语法示例,尽管原理对于所有循环和switch语句都是相同的:

label name: while condition {
    statements
}

下面的示例使用带有label的while循环的break和continue语句,为您在本章前面介绍的Snakes and Ladders游戏的改编版提供帮助。 这次,游戏有一个额外的规则:

  • 为了赢,您必须准确落在25号square上。

如果特定骰子掷骰使您超过25号方格,则必须再次掷骰,直到掷出准确的数字才能落在25号方格上。

游戏板与以前相同。

image

finalSquare,board,square和diceRoll的值以与之前相同的方式进行初始化:

let finalSquare = 25
var board = [Int](repeating: 0, count: finalSquare + 1)
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循环和switch语句来实现游戏的逻辑。 while循环具有一个称为gameLoop的语句标签,以指示它是Snakes and Ladders游戏的主要游戏循环。

while循环的条件是while方格== finalSquare,以反映您必须准确落在方格25上。

gameLoop: while square != finalSquare {
    diceRoll += 1
    if diceRoll == 7 { diceRoll = 1 }
    switch square + diceRoll {
    case finalSquare:
        // diceRoll will move us to the final square, so the game is over
        break gameLoop
    case let newSquare where newSquare > finalSquare:
        // diceRoll will move us beyond the final square, so roll again
        continue gameLoop
    default:
        // this is a valid move, so find out its effect
        square += diceRoll
        square += board[square]
    }
}
print("Game over!")

在每个循环开始时掷骰子。 循环不是立即移动播放器,而是使用switch语句来考虑移动结果并确定是否允许移动:

  • 如果掷骰子将玩家移至最后一个方块,则游戏结束。 break gameLoop语句将控制权转移到while循环之外的第一行代码,从而结束了游戏。
  • 如果掷骰子会将玩家移出最终方块,则该移动无效,玩家需要再次掷骰子。 Continue gameLoop语句结束当前的while循环迭代并开始循环的下一个迭代。
    在所有其他情况下,掷骰子都是有效的举动。
  • 玩家以diceRoll方块向前移动,游戏逻辑检查是否有蛇和梯zi。 然后循环结束,控制返回while条件,以决定是否需要再转一圈。

注意
如果上面的break语句不使用gameLoop标签,它将脱离switch语句,而不是while语句。 使用gameLoop标签可以清楚地表明应该终止哪个控制语句。

调用continue gameLoop跳转到循环的下一个迭代时,不一定必须使用gameLoop标签。 游戏中只有一个循环,因此,继续声明会影响哪个循环没有任何歧义。 但是,将gameLoop标签与continue语句一起使用不会有任何危害。 这样做与标签在break语句旁边的用法一致,并有助于使游戏的逻辑更易于阅读和理解。

5 提前退出(guard)

与if语句一样,guard语句根据表达式的布尔值执行语句。 您使用保护语句要求条件必须为true,以便执行保护语句之后的代码。 与if语句不同,guard语句始终具有else子句-如果条件不成立,则执行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(person: ["name": "John"])
// Prints "Hello John!"
// Prints "I hope the weather is nice near you."
greet(person: ["name": "Jane", "location": "Cupertino"])
// Prints "Hello Jane!"
// Prints "I hope the weather is nice in Cupertino."

如果满足了guard语句的条件,则在guard语句的右大括号之后代码将继续执行。 使用条件绑定(作为条件的一部分)为值分配了值的任何变量或常量,对于该guard语句所在的其余代码块均可用。

如果不满足该条件,则执行else分支内的代码。 该分支必须转移控制权以退出其中出现guard语句的代码块。 它可以通过控制传递语句(例如return,break,continue或throw)来执行此操作,也可以调用不返回的函数或方法,例如fatalError(_:file:line :)。

与使用if语句执行相同的检查相比,使用guard语句满足要求可以提高代码的可读性。 它使您可以编写通常执行的代码,而无需将其包装在else块中,并使处理违反要求的代码保持在要求旁边。

6 检查API可用性

Swift具有检查API可用性的内置支持,可确保您不会意外使用给定部署目标上不可用的API。

编译器使用SDK中的可用性信息来验证代码中使用的所有API在项目指定的部署目标上是否可用。 如果您尝试使用不可用的API,Swift会在编译时报告错误。

您可以在if或guard语句中使用可用性条件,以有条件地执行代码块,具体取决于要使用的API在运行时是否可用。 当编译器验证该代码块中的API可用时,将使用可用性条件中的信息。

if #available(iOS 10, macOS 10.12, *) {
    // Use iOS 10 APIs on iOS, and use macOS 10.12 APIs on macOS
} else {
    // Fall back to earlier iOS and macOS APIs
}

上面的可用性条件指定在iOS中,if语句的主体仅在iOS 10及更高版本中执行; 在macOS中,仅在macOS 10.12及更高版本中。 最后一个参数*是必需的,它指定在任何其他平台上,if的主体在目标所指定的最小部署目标上执行。

在一般情况下,可用性条件采用平台名称和版本的列表。 您使用平台名称,例如iOS,macOS,watchOS和tvOS,有关完整列表,请参阅Declaration Attributes。 除了指定主要版本号(例如iOS 8或macOS 10.10)之外,您还可以指定次要版本号(例如iOS 11.2.6和macOS 10.13.3)。

if #available(platform name version, ..., *) {
    statements to execute if the APIs are available
} else {
    fallback statements to execute if the APIs are unavailable
}

总结

这一章节学习的内容在这里做个总结:

  • for-in循环:可以直接声明变量来作为数组循环的返回值,或使用元组来作为字典循环的返回值,或者使用通配符_来忽略值;还可以使用...和..<来进行范围循环;还可以使用stride(from:to:by:)函数老跳过不需要的标记循环。
  • while循环:有两种while和repeat-while(类似C语言中的do-while)循环。
  • if条件语句,这个大家应该都会,这里不做赘述。
  • switch语句,swift里的和OC里的不一样,switch判断的值可以是任何类型,不像OC只能匹配整型,在swift里在一个case末尾不需要加break来跳出switch case,默认会直接结束switch case的判断;switch可以用间隔匹配的功能,如:case 1..<5:来匹配一个范围的值;还可以使用元组,如:case (0, 0)、case(_, 0)、case(0, _)、case(-2...2, -2...2)来匹配多个值;还可以使用值绑定,如:case (let x, 0)、case(0, let y)、case(let x, let y)来匹配一个或多个值;还可以使用where关键字来添加条件判断,如:case let (x, y) where x== y;还可以在case后加入多个模式,如:case "a","e","i"
  • 控制转移语句:continue、break,swift里的和OC的用法差不多;swift多出的功能是fallthrough和带标签语句。使用fallthrough来使switch语句能继续执行下一个case内的内容;使用带标签的语句,配合switch语句和break、contine判断带标签的变量是否满足条件,具体的实现可以看上面的代码。
  • guard语句,功能和if语句一样,不过用guard语句条件判断会更高效,且代码更简洁。
  • 检查API可用性,可配合if和guard语句使用,如:if #available(platform name version, ..., *)

上一章节:集合类型

下一章节:函数

参考文档:Swift - Control Flow

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容

  • 由苹果官网翻译得来fork自https://github.com/letsswift/The-Swift-Prog...
    佛祖拿屠刀阅读 279评论 1 3
  • Swift 提供了多种流程控制结构,包括可以多次执行任务的 while 循环,基于特定条件选择执行不同代码分支的 ...
    CoderLGL阅读 357评论 0 1
  • 多次执行任务的 while 循环 基于特定条件选择执行不同代码分支的 if、guard 和 switch 语句 控...
    Sunday_David阅读 211评论 0 0
  • 前言 Swift提供了多种流程控制结构,包括可以多次执行任务的while循环,基于特定条件选择执行不同代码分支的i...
    下页天阅读 221评论 0 0
  • Swift提供了多种流程控制结构,包括可以多次执行任务的while循环,基于特定条件基于特定条件选择执行不同代码分...
    edison0428阅读 1,235评论 0 0