The Swift Programming Language学习笔记六——控制流

控制流

Swift提供了类似C语言的流程控制结构,包括可以多次执行任务的forwhile循环,基于特定条件选择执行不同代码分支的ifguardswitch语句,还有控制流程跳转到其他代码的breakcontinue语句。

除了C语言里面传统的for循环,Swift还增加了for-in循环,用来更简单地遍历数组(array),字典(dictionary),区间(range),字符串(string)和其他序列类型。

Swift的switch语句比C语言中更加强大。在C语言中,如果某个case不小心漏写了break,这个case就会贯穿至下一个case,Swift无需写break,所以不会发生这种贯穿的情况。case还可以匹配更多的类型模式,包括区间匹配(range matching),元组(tuple)和特定类型的描述。switch的case语句中匹配的值可以是由case体内部临时的常量或者变量决定,也可以由where分句描述更复杂的匹配条件。

for循环

Swift提供了两种for循环形式

  • for-in循环对一个集合里面的每个元素执行一系列语句。
  • for循环,用来重复执行一系列语句直到达成特定条件达成,一般通过在每次循环完成后增加计数器的值来实现。

for-in

使用for-in循环来遍历一个集合里面的所有元素,例如由数字表示的区间、数组中的元素、字符串中的字符。

for i in 1...5 {
    print("\(i)",terminator: "-")
}

上面的i是一个每次循环遍历开始时被自动赋值的常量。这种情况下,i在使用前不需要声明,只需要将它包含在循环的声明中,就可以对其进行隐式声明,而无需使用let关键字声明

如果你不需要知道区间序列内每一项的值,你可以使用下划线(_)替代变量名来忽略对值的访问。

for _ in 0..<5 {
    print("Hello")
}

使用for-in遍历数组和字典。遍历字典时,字典的每项元素会以(key,value)元组的形式返回,你可以在for-in循环中使用显式的常量名称来解读(key,value)元组。

let a = [1,2,3]
for i in a {    // 注意:文档上说i是常量,但是貌似只能写var不能写let
    print(i)
}

let b = ["A": 1,"B": 2,"C": 3]
for var (k,v) in b {       // 注意:文档上说k和v是常量,但是貌似只能写var不能写let
    print("\(k) => \(v)") }

for

除了for-in循环,Swift提供使用条件判断和递增方法的标准C样式for循环。

for var i = 0; i < 3; ++i {     // 注意:此时的var必须写,而且不能写成let
    print(i)
}

在初始化表达式中声明的常量和变量(比如var i = 0)只在for循环的生命周期里有效。如果想在循环结束后访问i的值,你必须要在循环生命周期开始前声明i

while循环

这类循环适合使用在第一次迭代前迭代次数未知的情况下。Swift提供两种while循环形式:

  • while循环,每次在循环开始时计算条件是否符合
  • repeat-while循环,每次在循环结束时计算条件是否符合

while

while循环从计算单一条件开始。如果条件为true,会重复运行一系列语句,直到条件变为false

/** * 文档上的蛇和梯子的小游戏,也叫做滑道和梯子 */

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 square < finalSquare {
    // 掷骰子
    if ++diceRoll == 7 { diceRoll = 1 }
    // 根据点数移动
    square += diceRoll
    if square < board.count {
        // 如果玩家还在棋盘上,顺着梯子爬上去或者顺着蛇滑下去
        square += board[square]
    }
}
print("Game over!")
print("最终位置是\(square)。")     // 最终位置是27。

repeat-while

它和while的区别是在判断循环条件之前,先执行一次循环的代码块,然后重复循环直到条件为false。Swift语言的repeat-while循环和其他语言中的do-while循环是类似的。

/** * 文档上的蛇和梯子的小游戏,也叫做滑道和梯子 */

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 {
    // 顺着梯子爬上去或者顺着蛇滑下去
    square += board[square]
    // 掷骰子
    if ++diceRoll == 7 { diceRoll = 1 }
    // 根据点数移动
    square += diceRoll
} while square < finalSquare
print("Game over!")
print("最终位置是\(square)。")     // 最终位置是27。

可以看出,此时repeat-while表现得比while循环更好。repeat-while方式会在条件判断square没有超出后直接运行square += board[square],这种方式可以去掉while版本中的数组越界判断

条件语句

Swift提供两种类型的条件语句:if语句和switch语句。通常,当条件较为简单且可能的情况很少时,使用if语句。而switch语句更适用于条件较复杂、可能情况较多且需要用到模式匹配(pattern-matching)的情境。

if

  • if语句最简单的形式就是只包含一个条件,当且仅当该条件为true时,才执行相关代码。
  • if语句允许二选一,也就是当条件为false时,执行else语句。
  • 还可以把多个if语句链接在一起,此时,最后的else语句是可选的
let a = 10
if a < 10 {
    print("a < 10")
} else if a > 10 {
    print("a > 10")
} else {
    print("a == 10")
}

switch

switch语句会尝试把某个值与若干个模式(pattern)进行匹配。根据第一个匹配成功的模式,switch语句会执行对应的代码。当有可能的情况较多时,通常用switch语句替换if语句。

switch语句必须是完备的。这就是说,每一个可能的值都必须至少有一个case分支与之对应。在某些不可能涵盖所有值的情况下,你可以使用默认(default)分支满足该要求,这个默认分支必须在switch语句的最后面。switch语句不完备时,默认(default)分支必须有

不存在隐式的贯穿

与C语言和Objective-C中的switch语句不同,在Swift中,当匹配的case分支中的代码执行完毕后,程序会终止switch语句,而不会继续执行下一个case分支。这也就是说,不需要在case分支中显式地使用break语句。这使得switch语句更安全、更易用,也避免了因忘记写break语句而产生的错误。

虽然在Swift中break不是必须的,但你依然可以在case分支中的代码执行完毕前使用break跳出

每一个 case 分支都必须包含至少一条语句,避免了意外地从一个case分支贯穿到另外一个,使得代码更安全、也更直观。

一个case也可以包含多个模式,用逗号把它们分开(如果太长了也可以分行写)。

let a = 1
switch a {
// case 0:     // 每个case分支不能有空语句
case 1:
    print("1")
case 2,3:
    print("2 or 3")
default:    // 此时switch还不完备,因此必须有default分支
    print("Error")
}

区间匹配

case 分支的模式也可以是一个值的区间。

闭区间操作符(...)以及半开区间操作符(..<)功能被重载去返回IntervalTypeRange。一个区间可以决定他是否包含特定的元素,就像当匹配一个switch声明的case一样。区间是一个连续值的集合,可以用for-in语句遍历它。

let a = 1
switch a {
case 0...1:
    print("0 ~ 1")
case 2..<10:
    print("2 ~ 9")
default:    // 必须有default分支
    print("other")
}

元组

可以使用元组在同一个switch语句中测试多个值。元组中的元素可以是值,也可以是区间。另外,使用下划线(_)来匹配所有可能的值

let a = (1,1)
switch a {
case (0,0):
    print("在原点")
case (_,0):
    print("在x轴")
case (0,_):
    print("在y轴")
case (-1...1,-1...1):
    print("在格子里")   // 在格子里
default:
    print("在格子外")
}

let b = (0,0)
switch b {
case (0,0):
    print("在原点")    // 在原点,优先匹配上!
case (_,-1...1):
    print("在格子里")
default:
    print("在格子外")
}

不像C语言,Swift允许多个case匹配同一个值。实际上,在这个例子中,点(0,0)可以匹配所有四个case。但是,如果存在多个匹配,那么只会执行第一个被匹配到的case分支。考虑点(0,0)会首先匹配case(0,0),因此剩下的能够匹配(0,0)的case分支都会被忽视掉。

值绑定

case分支的模式允许将匹配的值绑定到一个临时的常量或变量,这些常量或变量在该case分支里就可以被引用了——这种行为被称为值绑定(value binding)。

let a = (0,1)
switch a {
case (let x,0):
    print("1、\(x)")
case (0,let x):
    print("2、\(x)")     // 21
case let (x,y):
    print("3、\(x),\(y)")    // 不需要default,因为已经完备
}

let b = (3,0)
switch b {
case (let x,0):        // 13
    print("1、\(x)")
case (0,let x):
    print("2、\(x)")
case let (x,\(y)")
}

let c = (23,-12)
switch c {
case (let x,\(y)")    // 323,-12
}

let d = (0,0)
switch d {
case (let x,0):
    print("1、\(x)")        // 10
case (0,\(y)")
}

这三个case都声明了常量xy的占位符,用于临时获取元组中的一个或两个值。一旦声明了这些临时的常量,它们就可以在其对应的case分支里引用。

第一个case将匹配一个纵坐标为0的点,并把这个点的横坐标赋给临时的常量x。类似的,第二个case将匹配一个横坐标为0的点,并把这个点的纵坐标赋给临时的常量y

这个switch语句不包含默认分支。这是因为最后一个case声明了一个可以匹配余下所有值的元组。这使得switch语句已经完备了!

这里x和y是常量,这是因为没有必要在其对应的case分支中修改它们的值。然而,它们也可以是变量,那么程序将会创建临时变量,并用相应的值初始化它。修改这些变量只会影响其对应的case分支。

where

case分支的模式可以使用where语句来判断额外的条件。

let a = (1,-1)
switch a {
case let (x,y) where x == y:
    print("x == y")
case let (x,y) where x == -y:
    print("x == -y")    // x == -y
case let (x,y):        // 已经完备,无需default
    print(a)
}

上面声明的常量xy被用作where语句的一部分,从而创建一个动态的过滤器(filter)。当且仅当where语句的条件为true时,匹配到的case分支才会被执行。

控制转移语句

控制转移语句改变你代码的执行顺序,通过它你可以实现代码的跳转。Swift有五种控制转移语句

  • continue
  • break
  • fallthrough
  • return(“函数”一节中介绍)
  • throw(“错误抛出”一节中介绍)

continue

continue语句告诉一个循环体立刻停止本次循环迭代,重新开始下次循环迭代。

在一个带有条件和递增的for循环体中,调用continue语句后,迭代增量仍然会被计算求值。循环体继续像往常一样工作,仅仅只是循环体中的执行代码会被跳过。

break

break语句会立刻结束整个控制流的执行。当你想要更早的结束一个switch代码块或者一个循环体时,你都可以使用break语句。

循环语句中的break

当在一个循环体中使用break时,会立刻中断该循环体的执行,然后跳转到表示循环体结束的大括号(})后的第一行代码。不会再有本次循环迭代的代码被执行,也不会再有下次的循环迭代产生。

switch语句中的break

当在一个switch代码块中使用break时,会立即中断该switch代码块的执行,并且跳转到表示switch代码块结束的大括号(})后的第一行代码。

这种特性可以被用来匹配或者忽略一个或多个分支。因为Swift的switch需要包含所有的分支而且不允许有为空的分支,有时为了使你的意图更明显,需要特意匹配或者忽略某个分支。那么当你想忽略某个分支时,可以在该分支内写上break语句。当那个分支被匹配到时,分支内的break语句立即结束switch代码块。

注意:当一个switch分支仅仅包含注释时,会被报编译时错误。注释不是代码语句而且也不能让switch分支达到被忽略的效果。你总是可以使用break来忽略某个分支。

fallthrough

如果你确实需要C风格的贯穿的特性,你可以在每个需要该特性的case分支中使用fallthrough`关键字。

fallthrough关键字不会检查它下一个将会落入执行的case中的匹配条件。fallthrough简单地使代码执行继续连接到下一个case中的执行代码,这和C语言标准中的switch语句特性是一样的。

let a = 3
switch a {
case 0, 1, 2:
    print("0 ~ 2")
case 3..<100:
    print("3 ~ 100")    // 3 ~ 100
    fallthrough
default:
    print("Over")       // Over
}

带标签的语句

在Swift中,你可以在循环体和switch代码块中嵌套循环体和switch代码块来创造复杂的控制流结构。然而,循环体和switch代码块两者都可以使用break语句来提前结束整个方法体。因此,显式地指明break语句想要终止的是哪个循环体或者switch代码块,会很有用。类似地,如果你有许多嵌套的循环体,显式指明continue语句想要影响哪一个循环体也会非常有用。

可以使用标签来标记一个循环体或者switch代码块,当使用break或者continue时,带上这个标签,可以控制该标签代表对象的中断或者执行。产生一个带标签的语句是通过在该语句的关键词的同一行前面放置一个标签,并且该标签后面还需带着一个冒号。

/** * 文档上的蛇和梯子的小游戏,也叫做滑道和梯子。 * 现在增加一个条件:为了获胜,你必须刚好落在第 25 个方块中。即如果某次掷骰子使你的移动超出第 25 个方块,你必须重新掷骰子,直到你掷出的骰子数刚好使你能落在第 25 个方块中。 */
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

gameLoop: while square != finalSquare {
    if ++diceRoll == 7 { diceRoll = 1 }
    switch square + diceRoll {
    case finalSquare:
        // 到达最后一个方块,游戏结束
        print(square + diceRoll)
        break gameLoop
    case let newSquare where newSquare > finalSquare:
        // 超出最后一个方块,再掷一次骰子
        print("square = \(square),diceRoll = \(diceRoll)")
        continue gameLoop
    default:
        // 本次移动有效
        square += diceRoll
        square += board[square]
    }
}
print("Game over!")
/* 过程是:23+1 => 24-8 => 16+2 => 18+3 => 21+4 => 25 square = 23,diceRoll = 4 square = 23,diceRoll = 5 square = 23,diceRoll = 6 25 Game over! */

提前退出

if语句一样,guard的执行取决于一个表达式的布尔值。我们可以使用guard语句来要求条件必须为真时,以执行guard语句后的代码。不同于if语句,一个guard语句总是有一个else分句,如果条件不为真则执行else分句中的代码。

如果guard语句的条件被满足,则在保护语句的封闭大括号结束后继续执行代码。任何使用了可选绑定作为条件的一部分并被分配了值的变量或常量对于剩下的保护语句出现的代码段是可用的。

如果条件不被满足,在else分支上的代码就会被执行。这个分支必须转移控制以退出guard语句出现的代码段它可以用控制转移语句如return,break,continue或者throw做这件事,或者调用一个不返回的方法或函数,例如fatalError()

相比于可以实现同样功能的if语句,按需使用guard语句会提升我们代码的可靠性。 它可以使你的代码连贯的被执行而不需要将它包在else块中,它可以使你处理违反要求的代码使其接近要求

let a = 10
/*
guard a < 5 else {      // error: 'guard' body may not fall through,consider using 'return' or 'break' to exit the scope
    print("a >= 5")
}
*/

while true {
    guard a < 5 else {
        print("a >= 5")     // a >= 5
        break   // 必须写break,否则报上面的错误
    }
}

print("...")

检测API可用性

Swift有检查API可用性的内置支持,这可以确保我们不会不小心地使用对于当前部署目标不可用的API。

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

我们使用一个可用性条件在一个ifguard语句中去有条件地执行一段代码,这取决于我们想要使用的API是否在运行时是可用的。编译器使用从可用性条件语句中获取的信息去验证在代码块中调用的API是否都可用。

if #available(iOS 9,OSX 10.10,*) {
    print("iOS 9+ or OS X 10.10+")      // 条件是或的关系,有一个平台满足就可以,表示指定了在iOS系统上,if段的代码仅会在iOS 9及更高版本的系统上执行;在OS X,仅会在OS X v10.10及更高版本的系统上执行。
} else {
    print("old version")
}

可用性条件指定了在iOS系统上,if段的代码仅会在iOS 9及更高版本的系统上执行;在OS X,仅会在OS X v10.10及更高版本的系统上执行。上面的最后一个参数*是必须写的,用于处理未来潜在的平台。可用性条件获取了一系列平台名字和版本。平台名字可以是iOSOSXwatchOS。除了特定的主板本号像iOS 8,我们可以指定较小的版本号像iOS 8.3以及OS X v10.10.3。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


软件简介:蓝湖辅助工具,减少移动端开发中控件属性的复制和粘贴.待开发的功能:1.支持自动生成约束2.开发设置页面3.做一个浏览器插件,支持不需要下载整个工程,可即时操作当前蓝湖浏览页面4.支持Flutter语言模板生成5.支持更多平台,如Sketch等6.支持用户自定义语言模板
现实生活中,我们听到的声音都是时间连续的,我们称为这种信号叫模拟信号。模拟信号需要进行数字化以后才能在计算机中使用。目前我们在计算机上进行音频播放都需要依赖于音频文件。那么音频文件如何生成的呢?音频文件的生成过程是将声音信息采样、量化和编码产生的数字信号的过程,我们人耳所能听到的声音频率范围为(20Hz~20KHz),因此音频文件格式的最大带宽是20KHZ。根据奈奎斯特的理论,音频文件的采样率一般在40~50KHZ之间。奈奎斯特采样定律,又称香农采样定律。...............
前言最近在B站上看到一个漂亮的仙女姐姐跳舞视频,循环看了亿遍又亿遍,久久不能离开!看着小仙紫姐姐的蹦迪视频,除了一键三连还能做什么?突发奇想,能不能把舞蹈视频转成代码舞呢?说干就干,今天就手把手教大家如何把跳舞视频转成代码舞,跟着仙女姐姐一起蹦起来~视频来源:【紫颜】见过仙女蹦迪吗 【千盏】一、核心功能设计总体来说,我们需要分为以下几步完成:从B站上把小姐姐的视频下载下来对视频进行截取GIF,把截取的GIF通过ASCII Animator进行ASCII字符转换把转换的字符gif根据每
【Android App】实战项目之仿抖音的短视频分享App(附源码和演示视频 超详细必看)
前言这一篇博客应该是我花时间最多的一次了,从2022年1月底至2022年4月底。我已经将这篇博客的内容写为论文,上传至arxiv:https://arxiv.org/pdf/2204.10160.pdf欢迎大家指出我论文中的问题,特别是语法与用词问题在github上,我也上传了完整的项目:https://github.com/Whiffe/Custom-ava-dataset_Custom-Spatio-Temporally-Action-Video-Dataset关于自定义ava数据集,也是后台
因为我既对接过session、cookie,也对接过JWT,今年因为工作需要也对接了gtoken的2个版本,对这方面的理解还算深入。尤其是看到官方文档评论区又小伙伴表示看不懂,所以做了这期视频内容出来:视频在这里:本期内容对应B站的开源视频因为涉及的知识点比较多,视频内容比较长。如果你觉得看视频浪费时间,可以直接阅读源码:goframe v2版本集成gtokengoframe v1版本集成gtokengoframe v2版本集成jwtgoframe v2版本session登录官方调用示例文档jwt和sess
【Android App】实战项目之仿微信的私信和群聊App(附源码和演示视频 超详细必看)
用Android Studio的VideoView组件实现简单的本地视频播放器。本文将讲解如何使用Android视频播放器VideoView组件来播放本地视频和网络视频,实现起来还是比较简单的。VideoView组件的作用与ImageView类似,只是ImageView用于显示图片,VideoView用于播放视频。...
采用MATLAB对正弦信号,语音信号进行生成、采样和内插恢复,利用MATLAB工具箱对混杂噪声的音频信号进行滤波
随着移动互联网、云端存储等技术的快速发展,包含丰富信息的音频数据呈现几何级速率增长。这些海量数据在为人工分析带来困难的同时,也为音频认知、创新学习研究提供了数据基础。在本节中,我们通过构建生成模型来生成音频序列文件,从而进一步加深对序列数据处理问题的了解。
基于yolov5+deepsort+slowfast算法的视频实时行为检测。1. yolov5实现目标检测,确定目标坐标 2. deepsort实现目标跟踪,持续标注目标坐标 3. slowfast实现动作识别,并给出置信率 4. 用框持续框住目标,并将动作类别以及置信度显示在框上
数字电子钟设计本文主要完成数字电子钟的以下功能1、计时功能(24小时)2、秒表功能(一个按键实现开始暂停,另一个按键实现清零功能)3、闹钟功能(设置闹钟以及到时响10秒)4、校时功能5、其他功能(清零、加速、星期、八位数码管显示等)前排提示:前面几篇文章介绍过的内容就不详细介绍了,可以看我专栏的前几篇文章。PS.工程文件放在最后面总体设计本次设计主要是在前一篇文章 数字电子钟基本功能的实现 的基础上改编而成的,主要结构不变,分频器将50MHz分为较低的频率备用;dig_select
1.进入官网下载OBS stdioOpen Broadcaster Software | OBS (obsproject.com)2.下载一个插件,拓展OBS的虚拟摄像头功能链接:OBS 虚拟摄像头插件.zip_免费高速下载|百度网盘-分享无限制 (baidu.com)提取码:6656--来自百度网盘超级会员V1的分享**注意**该插件必须下载但OBS的根目录(应该是自动匹配了的)3.打开OBS,选中虚拟摄像头选择启用在底部添加一段视频录制选择下面,进行录制.
Meta公司在9月29日首次推出一款人工智能系统模型:Make-A-Video,可以从给定的文字提示生成短视频。基于**文本到图像生成技术的最新进展**,该技术旨在实现文本到视频的生成,可以仅用几个单词或几行文本生成异想天开、独一无二的视频,将无限的想象力带入生活
音频信号叠加噪声及滤波一、前言二、信号分析及加噪三、滤波去噪四、总结一、前言之前一直对硬件上的内容比较关注,但是可能是因为硬件方面的东西可能真的是比较杂,而且需要渗透的东西太多了,所以学习进展比较缓慢。因为也很少有单纯的硬件学习研究,总是会伴随着各种理论需要硬件做支撑,所以还是想要慢慢接触理论学习。但是之前总找不到切入点,不知道从哪里开始,就一直拖着。最近稍微接触了一点信号处理,就用这个当作切入点,开始接触理论学习。二、信号分析及加噪信号处理选用了matlab做工具,选了一个最简单的语音信号处理方
腾讯云 TRTC 实时音视频服务体验,从认识 TRTC 到 TRTC 的开发实践,Demo 演示& IM 服务搭建。
音乐音频分类技术能够基于音乐内容为音乐添加类别标签,在音乐资源的高效组织、检索和推荐等相关方面的研究和应用具有重要意义。传统的音乐分类方法大量使用了人工设计的声学特征,特征的设计需要音乐领域的知识,不同分类任务的特征往往并不通用。深度学习的出现给更好地解决音乐分类问题提供了新的思路,本文对基于深度学习的音乐音频分类方法进行了研究。首先将音乐的音频信号转换成声谱作为统一表示,避免了手工选取特征存在的问题,然后基于一维卷积构建了一种音乐分类模型。
C++知识精讲16 | 井字棋游戏(配资源+视频)【赋源码,双人对战】
本文主要讲解如何在Java中,使用FFmpeg进行视频的帧读取,并最终合并成Gif动态图。
在本篇博文中,我们谈及了 Swift 中 some、any 关键字以及主关联类型(primary associated types)的前世今生,并由浅及深用简明的示例向大家讲解了它们之间的奥秘玄机。