2018-07-09

控制流

Swift提供了各种控制流程语句。这些包括while循环多次执行任务; ifguard以及switch基于特定条件执行不同分支代码的语句; 以及诸如此类的语句break以及continue将执行流程转移到代码中的另一个点。

迅速还提供了for- in循环,可以很容易地遍历数组,字典,范围,字符串和其它序列。

Swift的switch声明比许多C语言中的声明强大得多。个案可以匹配许多不同的模式,包括区间匹配,元组和特定类型的强制转换。switch案例中的匹配值可以绑定到临时常量或变量以用于案例的正文中,并且复杂的匹配条件可以用where每个案例的子句表示。

For-In循环

您可以使用for- in循环遍历序列,例如数组中的项目,数字范围或字符串中的字符。

此示例使用for- in循环来迭代在阵列中的项目:

  1. let names = ["Anna", "Alex", "Brian", "Jack"]
  2. for name in names {
  3. print("Hello, \(name)!")
  4. }
  5. // Hello, Anna!
  6. // Hello, Alex!
  7. // Hello, Brian!
  8. // Hello, Jack!

您也可以迭代字典来访问其键值对。(key, value)当字典迭代时,字典中的每一项都作为元组返回,并且可以将(key, value)元组的成员分解为明确命名的常量,以便在for- in循环体内使用。在下面的代码示例中,字典的键被分解为一个常量animalName,字典的值被分解为一个常量legCount

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

a的内容Dictionary本质上是无序的,迭代它们并不能保证它们被检索的顺序。特别是,将项插入到a Dictionary的顺序不会定义它们被迭代的顺序。有关数组和字典的更多信息,请参阅集合类型

您也可以使用for- in数字范围的循环。此示例在五次表中打印前几个条目:

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

正在迭代的序列是一系列从0 1到0 的数字5,如使用闭范围运算符(...)所示。该值index设置为范围(1)中的第一个数字,并执行循环内的语句。在这种情况下,循环仅包含一条语句,该语句从当前值的五次表中输入条目index。语句执行后,值index更新为包含范围(2)中的第二个值,并print(_:separator:terminator:)再次调用该函数。这个过程一直持续到范围的结束。

在上面的例子中,index是一个常量,其值在循环的每次迭代开始时自动设置。因此,index在使用之前不必声明。它仅仅通过包含在循环声明中而被隐式声明,而不需要let声明关键字。

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

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

上面的例子计算了一个数字与另一个数字的值(在这种情况下,3是幂的10)。它使用以开始和结束的封闭范围乘以1(即,3的幂0)的起始值3十倍。对于这种计算,每次循环中的单个计数器值都是不必要的 - 代码只需执行正确的循环次数。用于代替循环变量的下划线字符()会导致单个值被忽略,并且在循环的每次迭代过程中都不提供对当前值的访问。 1``10``_

在某些情况下,您可能不想使用封闭范围,其中包括两个端点。考虑在表盘上每刻刻绘制刻度标记。你想600分钟开始绘制刻度线。使用半开范围运算符(..<)来包含下界但不包含上界。有关范围的更多信息,请参阅范围操作员

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

有些用户可能需要在他们的用户界面中减少刻度线 他们可能会更喜欢每5分钟一个标记。使用该stride(from:to:by:)功能跳过不需要的标记。

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

封闭的范围也可以使用,stride(from:through:by:)取而代之:

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

虽然循环

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

  • while 评估每次通过循环开始时的状态。

  • repeat- while评估每次通过循环结束时的状态。

一个while循环开始通过评估一个条件。如果条件true成立,则重复一组陈述直到条件变为止false

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

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">而 条件 {</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 声明</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">}</pre>

这个例子演示了一个简单的Snakes和Ladders(也称为Chutes and Ladders)游戏:

<figure class="figure" style="font-size: 0.8rem;">[图片上传失败...(image-4a71f4-1531103424282)]

</figure>

游戏规则如下:

  • 该委员会有25个广场,目的是降落在25平方或以上。

  • 玩家的起始方格是“正方形零点”,就在董事会的左下角。

  • 每回合,你掷出一个六面骰子,并按照上面虚线箭头指示的水平路径移动该数量的正方形。

  • 如果你的回合在梯子的底部结束,那么你在梯子上移动。

  • 如果你的回合结束于蛇的头部,那么你会沿着那条蛇移动。

游戏板由一系列Int值表示。其大小基于一个常量finalSquare,用于初始化数组,并在稍后的示例中检查胜利条件。因为玩家从棋盘开始,在“方块零点”上,棋盘以26个零Int值进行初始化,而不是25个。

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

然后一些正方形设置为蛇和梯子的更具体的值。带梯子底座的方块有一个正数,可以将您移动到棋盘上,而带有蛇头的方块有负数可以让您退回棋盘。

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

方3包含移动你到平方11.要表示这样的阶梯的底部,board[03]是等于+08,这相当于一个整数值8(之间的差311)。为了对齐值和语句,一元加运算符(+i)与一元减运算符(-i)一起明确使用,低于10用零填充的数字。(风格技术不是绝对必要的,但它们导致整洁的代码。)

  1. var square = 0
  2. var diceRoll = 0
  3. while square < finalSquare {
  4. // roll the dice
  5. diceRoll += 1
  6. if diceRoll == 7 { diceRoll = 1 }
  7. // move by the rolled amount
  8. square += diceRoll
  9. if square < board.count {
  10. // if we're still on the board, move up or down for a snake or a ladder
  11. square += board[square]
  12. }
  13. }
  14. print("Game over!")

上面的例子使用了非常简单的方法来掷骰子。不是生成一个随机数,而是从一个diceRoll值开始0。每次while循环时,diceRoll都会加1,然后检查它是否变得过大。只要这个返回值相等7,骰子滚动就会变得太大,并被重置为一个值1。结果是一个序列diceRoll,始终是价值观12345612等。

掷骰子后,玩家前进diceRoll广场。掷骰子有可能将玩家移动到方块25以外,在这种情况下游戏结束。为了处理这种情况,代码检查的结果square小于board数组的count属性。如果square有效,则存储的值将board[square]被添加到当前square值以将播放器向上或向下移动任何梯子或蛇。

注意

如果未执行此检查,则board[square]可能尝试访问board数组边界外的值,这会触发运行时错误。

然后当前while循环执行结束,检查循环的条件以查看是否应该再次执行循环。如果玩家已经移动或超出方块数25,则回路的状态评估为false游戏结束。

一个while循环是在这种情况下,适当的,因为游戏的长度是不是在开始明确while循环。相反,循环被执行直到满足特定的条件。

重复,虽然

while循环 的另一种变体(称为repeat- while循环)考虑循环条件之前首先执行一次循环块。然后它继续重复循环直到条件成立false

注意

repeat- while环斯威夫特类似于一个do- while循环其他语言。

这里有一个的一般形式repeat- while循环:

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">重复 {</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 声明</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">} 而 条件</pre>

这里的蛇和梯子例子再次,写成repeat- while循环而不是一个while循环。的值finalSquareboardsquare,和diceRoll以完全相同的方式初始化为一个while循环。

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

在这个版本的游戏中,循环中的第一个动作是检查梯子或蛇。棋盘上没有任何梯子将球员直接拉到25平方,所以不可能通过上梯来赢得比赛。因此,检查一条蛇或梯子是循环中的第一个动作是安全的。

在比赛开始时,玩家处于“平方零点”。board[0]总是平等0并且没有效果。

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

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

循环的条件(while square < finalSquare)与以前相同,但是这次它不会被计算,直到循环的第一次运行结束repeat- while循环的结构比while上例中的循环更适合这个游戏。在repeat- while上面的循环,square += board[square]始终执行后立即循环的while条件证实,square仍然是在黑板上。这种行为消除了在while前面描述的游戏的循环版本中看到的数组边界检查的需要。

条件声明

根据特定条件执行不同的代码通常很有用。发生错误时可能需要运行一段代码,或者当值变得过高或过低时显示消息。要做到这一点,你需要有条件的部分代码。

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

如果

最简单的形式是if声明只有一个if条件。它只有在条件满足时才执行一组语句true

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

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

if语句可以提供一组替代语句,称为else子句,用于if条件满足的情况false。这些陈述由else关键字指示。

  1. temperatureInFahrenheit = 40
  2. if temperatureInFahrenheit <= 32 {
  3. print("It's very cold. Consider wearing a scarf.")
  4. } else {
  5. print("It's not that cold. Wear a t-shirt.")
  6. }
  7. // Prints "It's not that cold. Wear a t-shirt."

这两个分支之一总是被执行。由于温度已经升高到40华氏度,所以不再建议穿围巾,所以else分支被触发。

您可以将多个if语句链接在一起考虑附加的子句。

  1. temperatureInFahrenheit = 90
  2. if temperatureInFahrenheit <= 32 {
  3. print("It's very cold. Consider wearing a scarf.")
  4. } else if temperatureInFahrenheit >= 86 {
  5. print("It's really warm. Don't forget to wear sunscreen.")
  6. } else {
  7. print("It's not that cold. Wear a t-shirt.")
  8. }
  9. // Prints "It's really warm. Don't forget to wear sunscreen."

在这里,增加了一个额外的if声明以应对特别温暖的气温。最后的else条款仍然存在,并且它对任何既不太热也不太冷的温度打印响应。

else然而, 最终条款是可选的,如果条件集合不需要完整,则可以排除该条款。

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

因为温度不太冷太热以触发ifelse if条件下,印刷的任何消息。

开关

一个switch声明会考虑一个值并将其与几种可能的匹配模式进行比较。然后根据成功匹配的第一个模式执行适当的代码块。一个switch语句提供的另一种if用于应对多种潜在状态的语句。

以最简单的形式,一个switch语句将一个值与一个或多个相同类型的值进行比较。

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">切换 一些价值来考虑 {</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">案例 值1:</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 回应价值1</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">案例 值2,</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 值3:</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 回应价值2或3</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">默认值:</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 否则,做别的事情</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">}</pre>

每个switch陈述包含多个可能的案例,每个案例都以case关键字开头。除了与特定值进行比较外,Swift还为每种情况提供了几种方式来指定更复杂的匹配模式。这些选项将在本章稍后介绍。

if声明的主体一样,每个声明case都是代码执行的独立分支。该switch声明确定应该选择哪个分支。此过程称为打开正在考虑的值。

每个switch陈述都必须是详尽的。也就是说,所考虑类型的每个可能值都必须与其中一种switch情况相匹配。如果不适合为每个可能的值提供一个案例,则可以定义一个默认案例来覆盖任何未明确解决的值。此默认情况由default关键字指示,并且必须始终显示最后一个。

本示例使用switch语句来考虑一个名为的小写字符someCharacter

  1. let someCharacter: Character = "z"
  2. switch someCharacter {
  3. case "a":
  4. print("The first letter of the alphabet")
  5. case "z":
  6. print("The last letter of the alphabet")
  7. default:
  8. print("Some other character")
  9. }
  10. // Prints "The last letter of the alphabet"

switch语句的第一种情况相匹配的英文字母的第一个字母,a和它的第二种情况下的最后一个字母匹配,z。因为switch每个可能的字符都必须有一个字符,而不仅仅是每个字母字符,所以该switch语句使用一个default大小写来匹配除aand 之外的所有字符z。该条款确保该switch声明是详尽无遗的。

没有隐含的穿透

switchC和Objective-C中的switch语句相比,Swift中的语句不会落在每个案例的底部,默认情况下是下一个。相反,switch只要第一个匹配switch案例完成,整个语句就会结束执行,而不需要明确的break语句。这使得switch声明比C中的声明更安全和更易于使用,并避免switch错误地执行多个案例。

注意

虽然break在Swift中不是必需的,但您可以使用break语句来匹配和忽略特定的案例,或者在该案例已经完成执行之前突破匹配的案例。有关详细信息,请参阅切换语句中的中断

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

  1. let anotherCharacter: Character = "a"
  2. switch anotherCharacter {
  3. case "a": // Invalid, the case has an empty body
  4. case "A":
  5. print("The letter A")
  6. default:
  7. print("Not the letter A")
  8. }
  9. // This will report a compile-time error.

switchC中的陈述不同,这个switch陈述不符合两者"a""A"。相反,它会报告case "a":不包含任何可执行语句的编译时错误。这种方法避免了从一种情况到另一种情况的意外损失,并且使得安全代码的意图更加清晰。

为了使switch与匹配二者的单一情况下"a""A",这两个值组合成的化合物的情况下,用逗号分隔的值。

  1. let anotherCharacter: Character = "a"
  2. switch anotherCharacter {
  3. case "a", "A":
  4. print("The letter A")
  5. default:
  6. print("Not the letter A")
  7. }
  8. // Prints "The letter A"

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

注意

要在特定switch情况结束时明确贯彻,请使用fallthrough关键字,如Fallthrough中所述

间隔匹配

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

  1. let approximateCount = 62
  2. let countedThings = "moons orbiting Saturn"
  3. let naturalCount: String
  4. switch approximateCount {
  5. case 0:
  6. naturalCount = "no"
  7. case 1..<5:
  8. naturalCount = "a few"
  9. case 5..<12:
  10. naturalCount = "several"
  11. case 12..<100:
  12. naturalCount = "dozens of"
  13. case 100..<1000:
  14. naturalCount = "hundreds of"
  15. default:
  16. naturalCount = "many"
  17. }
  18. print("There are \(naturalCount) \(countedThings).")
  19. // Prints "There are dozens of moons orbiting Saturn."

在上面的例子中,approximateCountswitch声明中进行评估。每个case将该值与一个数字或区间进行比较。因为approximateCount落在12和100之间naturalCount的值被分配了值"dozens of",并且执行从switch语句中转出。

元组

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

下面的例子用一个(x,y)点表示,表示为一个简单的类型元组(Int, Int),并将它分类到示例后面的图上。

  1. let somePoint = (1, 1)
  2. switch somePoint {
  3. case (0, 0):
  4. print("\(somePoint) is at the origin")
  5. case (_, 0):
  6. print("\(somePoint) is on the x-axis")
  7. case (0, _):
  8. print("\(somePoint) is on the y-axis")
  9. case (-2...2, -2...2):
  10. print("\(somePoint) is inside the box")
  11. default:
  12. print("\(somePoint) is outside of the box")
  13. }
  14. // Prints "(1, 1) is inside the box"

<figure class="figure" style="font-size: 0.8rem;">[图片上传失败...(image-97e27a-1531103424277)]

</figure>

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

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

价值绑定

一个switch案例可以将它匹配的值或值命名为临时常量或变量,以便在案例正文中使用。这种行为被称为值绑定,因为值被绑定到案例正文中的临时常量或变量。

下面的例子将一个(x,y)点表示为一个类型的元组(Int, Int),并将它分类到下面的图表中:

  1. let anotherPoint = (2, 0)
  2. switch anotherPoint {
  3. case (let x, 0):
  4. print("on the x-axis with an x value of \(x)")
  5. case (0, let y):
  6. print("on the y-axis with a y value of \(y)")
  7. case let (x, y):
  8. print("somewhere else at (\(x), \(y))")
  9. }
  10. // Prints "on the x-axis with an x value of 2"

<figure class="figure" style="font-size: 0.8rem;">[图片上传失败...(image-215e58-1531103424277)]

</figure>

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

这三种switch情况下,声明占位符常量xy,暂时采取在一个或两个元组值的anotherPoint。第一种情况,case (let x, 0)匹配任何一个y值为0的点,并将该点的x值赋给临时常量x。类似地,第二种情况,case (0, let y)匹配x值为的任意点,0并将该点的y值分配给临时常量y

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

这个switch声明没有default案件。最后一种情况是case let (x, y),声明一个可以匹配任何值的两个占位符常量的元组。因为anotherPoint它总是一个包含两个值的元组,所以这种情况会匹配所有可能的剩余值,并且default不需要一个例子来使switch语句穷举。

哪里

一个switch案例可以使用一个where条款来检查附加条件。

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

  1. let yetAnotherPoint = (1, -1)
  2. switch yetAnotherPoint {
  3. case let (x, y) where x == y:
  4. print("(\(x), \(y)) is on the line x == y")
  5. case let (x, y) where x == -y:
  6. print("(\(x), \(y)) is on the line x == -y")
  7. case let (x, y):
  8. print("(\(x), \(y)) is just some arbitrary point")
  9. }
  10. // Prints "(1, -1) is on the line x == -y"

<figure class="figure" style="font-size: 0.8rem;">[图片上传失败...(image-7639fa-1531103424277)]

</figure>

switch声明确定该点是否在绿色对角线上x == y,其中,在紫色对角线上x == -y,还是两者都不是。

这三种switch情况下,声明占位符常量xy,暂时采取从两元组值yetAnotherPoint。这些常量用作where子句的一部分,以创建动态过滤器。只有当条款的条件评估为该switch值时,该案例才与当前值匹配。point``where``true

和前面的例子一样,最后一个案例匹配所有可能的剩余值,因此default不需要一个案例来使switch语句穷举。

复合案例

共享相同主体的多个开关盒可以通过case在各个图案之间以逗号之间写入多个图案来组合。如果任何模式匹配,则认为该情况匹配。如果列表很长,模式可以写入多行。例如:

  1. let someCharacter: Character = "e"
  2. switch someCharacter {
  3. case "a", "e", "i", "o", "u":
  4. print("\(someCharacter) is a vowel")
  5. case "b", "c", "d", "f", "g", "h", "j", "k", "l", "m",
  6. "n", "p", "q", "r", "s", "t", "v", "w", "x", "y", "z":
  7. print("\(someCharacter) is a consonant")
  8. default:
  9. print("\(someCharacter) is not a vowel or a consonant")
  10. }
  11. // Prints "e is a vowel"

switch语句的第一种情况相匹配的英语所有五个元音字母小写。同样,它的第二个案例匹配所有小写英语辅音。最后,该default案件与任何其他角色相匹配。

复合个案也可以包含值绑定。复合大小写的所有模式都必须包含相同的一组数值绑定,并且每个绑定必须从复合大小写中的所有模式中获取相同类型的值。这可以确保:无论案件的哪个部分匹配,案件正文中的代码都可以访问绑定的值,并且该值始终具有相同的类型。

  1. let stillAnotherPoint = (9, 0)
  2. switch stillAnotherPoint {
  3. case (let distance, 0), (0, let distance):
  4. print("On an axis, \(distance) from the origin")
  5. default:
  6. print("Not on an axis")
  7. }
  8. // Prints "On an axis, 9 from the origin"

case上面有两个模式:(let distance, 0)在x轴的匹配点和(0, let distance)在y轴一致点。两种模式都包含一个绑定,distance并且distance在这两种模式中都是一个整数 - 这意味着case可以总是访问某个值的代码distance

控制转移语句

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

  • continue

  • break

  • fallthrough

  • return

  • throw

continuebreakfallthrough语句描述如下。该return语句在函数中进行throw描述,声明在使用投掷函数传播错误中进行了描述。

继续

continue语句告诉循环停止它正在做的事情,并在循环的下一次迭代开始时重新开始。它说“我已经完成了当前的循环迭代”而不会完全离开循环。

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

  1. let puzzleInput = "great minds think alike"
  2. var puzzleOutput = ""
  3. let charactersToRemove: [Character] = ["a", "e", "i", "o", "u", " "]
  4. for character in puzzleInput {
  5. if charactersToRemove.contains(character) {
  6. continue
  7. } else {
  8. puzzleOutput.append(character)
  9. }
  10. }
  11. print(puzzleOutput)
  12. // Prints "grtmndsthnklk"

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

打破

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

循环陈述中断

在循环语句中使用时,break立即结束循环的执行,并在控制循环的大括号(})后将控制权转交给代码。不会执行循环当前迭代的其他代码,也不会开始循环的进一步迭代。

断开开关语句

switch语句内部使用时,break会导致switch语句立即结束其执行,并在控制switch语句的大括号(})结束后将控制权转交给代码。

此行为可用于匹配和忽略switch语句中的一个或多个个案。因为斯威夫特的switch陈述是详尽无遗的,并且不允许出现空案,所以有时需要刻意地匹配和忽略案件,以便明确地表达你的意图。您可以通过将该break陈述写为您想要忽略的整个案例来做到这一点。当该switch语句与该语句匹配时,该break语句内部的switch语句立即结束该语句的执行。

注意

一个switch仅包含注释的情况下被报告为编译时错误。评论不是陈述,也不会导致switch案件被忽略。始终使用break语句来忽略一个switch案例。

以下示例将打开一个Character值并确定它是否以四种语言之一表示数字符号。为简洁起见,在一个switch案例中涵盖了多个值。

  1. let numberSymbol: Character = "三" // Chinese symbol for the number 3
  2. var possibleIntegerValue: Int?
  3. switch numberSymbol {
  4. case "1", "١", "一", "๑":
  5. possibleIntegerValue = 1
  6. case "2", "٢", "二", "๒":
  7. possibleIntegerValue = 2
  8. case "3", "٣", "三", "๓":
  9. possibleIntegerValue = 3
  10. case "4", "٤", "四", "๔":
  11. possibleIntegerValue = 4
  12. default:
  13. break
  14. }
  15. if let integerValue = possibleIntegerValue {
  16. print("The integer value of \(numberSymbol) is \(integerValue).")
  17. } else {
  18. print("An integer value could not be found for \(numberSymbol).")
  19. }
  20. // Prints "The integer value of 三 is 3."

本示例检查numberSymbol以确定它是否是数字拉丁,阿拉伯语,中国,泰国或符号14。如果找到匹配项,则switch声明的其中一个案例将设置一个可选Int?变量,并将其调用possibleIntegerValue为适当的整数值。

在之后switch的语句执行完成后,该示例使用可选的结合来确定值是否被发现。该possibleIntegerValue变量nil由于是一个可选类型而具有隐式初始值,因此只有possibleIntegerValueswitch语句前四种情况之一中设置为实际值时,可选绑定才会成功。

因为Character在上面的例子中列出每个可能的值是不现实的,所以一个defaultcase处理任何不匹配的字符。这种default情况不需要执行任何操作,因此它是以单一break语句作为其主体编写的。只要default匹配,该break语句就会结束switch语句的执行,代码继续执行if let

下通

在Swift中,switch陈述不会落在每个案例的底部,并进入下一个案例。也就是说,switch只要第一个匹配的案例完成,整个语句就会完成它的执行。相比之下,C要求您break在每个switch案例的末尾插入一个明确的陈述以防止漏洞。避免默认的延迟意味着Swift switch语句比C语言中的对应语言更加简洁和可预测,因此避免switch了错误地执行多个事件。

如果您需要C风格的贯穿行为,您可以根据具体情况选择加入该fallthrough关键字的行为。下面的例子fallthrough用来创建一个数字的文本描述。

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

这个例子声明了一个新的String变量,description并为它分配一个初始值。该函数然后考虑integerToDescribe使用switch语句的值。如果值integerToDescribe是列表中的一个素数,则该函数将文本附加到末尾description,以注意该数字是素数。然后它使用fallthrough关键字“陷入” default情况。该default案例在描述的末尾添加了一些额外的文本,并且switch声明已完成。

除非integerToDescribe是已知素数列表中的值,否则根本不匹配第一种switch情况。因为没有其他特定的情况,integerToDescribe所以与default案例相匹配。

在之后switch的语句执行完毕,数量的描述是使用打印print(_:separator:terminator:)功能。在本例中,该号码5被正确识别为质数。

注意

fallthrough关键字不检查案件的switch情况下,它导致执行陷入。的fallthrough关键字简单地使代码执行直接移动到下一个的情况下(或内的语句default的情况下)嵌段,在C的标准switch语句的行为。

标记语句

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

为了实现这些目标,您可以用语句标签标记循环语句或条件语句。使用条件语句,您可以在语句中使用语句标签break来结束标记语句的执行。使用循环语句,可以使用带有breakor continue语句的语句标签来结束或继续执行带标签的语句。

带标签的语句通过将标签放置在与语句的介绍人关键字相同的行上,后跟冒号。下面是while循环语法的一个例子,虽然所有循环和switch语句的原理是相同的:

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">标签名称:while 条件 {</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 声明</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">}</pre>

下面的示例使用break,并continue声明与标记while的改编版环蛇和梯子,你在本章前面看到的比赛。这一次,游戏有一个额外的规则:

  • 要赢得比赛,你必须完全登上25号广场。

如果一个特定的掷骰子会使你超过25平方,你必须再次掷出,直到你掷出25号方所需的确切数量。

游戏板和以前一样。

<figure class="figure" style="font-size: 0.8rem;">[图片上传失败...(image-f38ed6-1531103424280)]

</figure>

的值finalSquareboardsquare,并diceRoll以同样的方式为前初始化:

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

该版本的游戏使用while循环和switch声明来实现游戏的逻辑。该while循环有一个声明标签gameLoop,表明它是Snakes and Ladders游戏的主要游戏循环。

while循环的条件为while square != finalSquare,以反映您必须准确降落在广场25。

  1. gameLoop: while square != finalSquare {
  2. diceRoll += 1
  3. if diceRoll == 7 { diceRoll = 1 }
  4. switch square + diceRoll {
  5. case finalSquare:
  6. // diceRoll will move us to the final square, so the game is over
  7. break gameLoop
  8. case let newSquare where newSquare > finalSquare:
  9. // diceRoll will move us beyond the final square, so roll again
  10. continue gameLoop
  11. default:
  12. // this is a valid move, so find out its effect
  13. square += diceRoll
  14. square += board[square]
  15. }
  16. }
  17. print("Game over!")

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

  • 如果骰子掷出将玩家移动到最后一个方格中,则游戏结束。该break gameLoop语句将控制转移到while循环外部的第一行代码,从而结束游戏。

  • 如果掷骰子会将玩家移动到最后一个方格以外,则该移动无效,玩家需要再次掷骰。该continue gameLoop语句结束当前while循环迭代并开始循环的下一次迭代。

  • 在所有其他情况下,掷骰子是一个有效的举措。玩家通过diceRoll方块向前移动,游戏逻辑检查任何蛇和梯子。循环结束,并且控制返回到该while条件以决定是否需要另一转。

注意

如果break上面的陈述没有使用gameLoop标签,它将打破switch陈述,而不是while陈述。使用gameLoop标签可以清楚地知道应该终止哪个控制语句。

gameLoop调用continue gameLoop跳转到循环的下一次迭代时 ,使用标签并不是必须的。游戏中只有一个循环,因此continue语句将影响到哪个循环并不含糊。但是,gameLoopcontinue声明中使用标签没有任何伤害。这样做与标签在break声明旁边的使用保持一致,并有助于使游戏的逻辑更加清晰,便于阅读和理解。

提前退出

guard语句,像一个if语句,执行根据表达式的布尔值的语句。您使用guard语句来要求条件必须为真,以便guard执行语句后的代码。与if声明不同,guard声明总是有一个else子句 - else如果条件不正确,子句中的代码将被执行。

  1. func greet(person: [String: String]) {

  2. guard let name = person["name"] else {

  3. return

  4. }

  5. print("Hello \(name)!")

  6. guard let location = person["location"] else {

  7. print("I hope the weather is nice near you.")

  8. return

  9. }

  10. print("I hope the weather is nice in \(location).")

  11. }

  12. greet(person: ["name": "John"])

  13. // Prints "Hello John!"

  14. // Prints "I hope the weather is nice near you."

  15. greet(person: ["name": "Jane", "location": "Cupertino"])

  16. // Prints "Hello Jane!"

  17. // Prints "I hope the weather is nice in Cupertino."

如果guard语句的条件满足,代码将在guard语句的大括号之后继续执行。在guard语句出现的代码块的其余部分,可以使用任何使用可选绑定作为条件一部分赋值的变量或常量。

如果不满足该条件,else则执行分支内的代码。该分支必须传输控制以退出guard语句出现的代码块。它可以做到这一点与控制权转移的语句,如returnbreakcontinue,或者throw,也可以调用一个函数或方法不返回,如fatalError(_:file:line:)

与使用guard语句进行相同的检查相比,使用需求语句可以提高代码的可读性if。它允许您编写通常执行的代码,而无需将其包装在一个else块中,并且可以让代码处理违反要求的代码。

检查API可用性

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

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

您使用的可靠性条件中的一个ifguard语句来有条件地执行代码块,这取决于你想使用的API是否在运行时可用。编译器在验证该代码块中的API可用时使用可用性条件中的信息。

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

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

通用形式中,可用性条件采用平台名称和版本的列表。您可以使用平台的名称,如iOSmacOSwatchOS,和tvOS-对于完整列表,请参阅声明属性。除了指定主要版本号(如iOS 8或MacOS 10.10)之外,您还可以指定次要版本号,如iOS 11.2.6和macOS 10.13.3。

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">如果 #available(平台名称 版本...,*){</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 语句在API可用时执行</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">} else {</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;"> 如果API不可用,则执行回退语句</pre>

  • <pre class="code-voice" style="margin: 8px 0px; font-size: 0.85rem;">}</pre>

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

推荐阅读更多精彩内容

  • 53.计算字符 在字符串中获取字符值的数量, 可以使用字符串字符属性中的计数属性: let unusualMena...
    无沣阅读 1,090评论 0 4
  • 这是16年5月份编辑的一份比较杂乱适合自己观看的学习记录文档,今天18年5月份再次想写文章,发现简书还为我保存起的...
    Jenaral阅读 2,746评论 2 9
  • 控制流 for循环 for-in for while循环 while repeat-while 条件语句 if s...
    zhenyu54阅读 893评论 0 0
  • Swift提供了多种控制流声明。包括while循环来多次执行一个任务;if,guard和switch声明来根据确定...
    BoomLee阅读 1,944评论 0 3
  • 查理是一个普通的中学生,他因为见证着一件恶作剧但又不想出卖朋友,于是面临着一道艰难的选择题——要么坦白,要么被学校...
    花生樹阅读 185评论 0 0