iOS 学习 Swift 教程- 2.2 基本运算符(Basic Operators)

前言:
本篇文章的目的,在于记录学习过程,敦促自己,方便查看。

练习工具:Playground
学习网站: swift51

本页内容包括:

运算符是检查、改变、合并值的特殊符号或短语。例如,加号(+)将两个数相加(如 let i = 1 + 2)。更复杂的运算例子包括逻辑与运算符 &&(如 if enteredDoorCode && passedRetinaScan)

Swift支持大部分标准 C 语言的运算符,且改进许多特性来减少常规编码错误。如:赋值符(=)不返回值,以防止把想要判断相等运算符(==)的地方写成赋值符导致的错误。算术运算符(+-*/%等)会检测并不允许值溢出,以此来避免保存变量时由于变量大于或小于其类型所能承载的范围时导致的异常结果。当然允许你使用 Swift 的溢出运算符来实现溢出。详情参见溢出运算符

Swift 还提供了 C语言没有的区间运算符,例如 a..<ba...b,这方便我们表达一个区间内的数值。

本章节只描述了 Swift 中的基本运算符,高级运算符这章会包含Swift中的高级运算符,及如何自定义运算符,及如何进行自定义类型的运算符重载。

术语

运算符分为一元二元三元运算符:

  • 一元运算符对单一操作对象操作(如 -a)。一元运算符分前置运算符和后置运算符,前置运算符需紧跟在操作对象之前(如 !b),后置运算符需紧跟在操作对象之后(如 c!)

  • 二元运算符操作两个操作对象(如 2 + 3),是中置的,因为它们出现在两个操作对象之间。

  • 三元运算符操作三个操作对象,和C 语言一样,Swift只有一个三元运算符,就是三目运算符(a ? b : c)

受运算符影响的值叫操作数,在表达式 1 + 2 中,加号 + 是二元运算符,它的两个操作数是值12

赋值运算符

赋值运算符(a = b) ,表示用b 的值来初始化或更新a 的值:

 let b = 10
 var a = 5
 a = b
 // a 现在等于 10

如果赋值的右边是一个多元组 ,它的元素可以马上被分解成多个常量或变量:

 let (x, y) = (1, 2)
 // 现在 x 等于 1,y 等于 2

C 语言和 Objective-C不同,Swift 的赋值操作并不返回任何值。所以以下代码是错误的:

 if x = y {
     // 此句错误, 因为 x = y 并不返回任何值
 }

这个特性使你无法把(==)错写成(=),由于if x = y 是错误代码,Swift能帮你避免此类错误发生。

算术运算符

Swift中所有数值类型都支持了基本的四则算术运算符

  • 加法(+)

  • 减法(-)

  • 乘法(*)

  • 除法(/)

    1 + 2       // 等于 3
    5 - 3       // 等于 2
    2 * 3       // 等于 6
    10.0 / 2.5  // 等于 4.0
    

C 语言和 Objective-C不同的是,Swift默认情况下不允许在数值运算中出现溢出情况。但是你可以使用Swift 的溢出运算符来实现溢出运算(如 a &+ b)。详情参见溢出运算符

加法运算符也可用于 String 的拼接:

 hello, " + "world"  // 等于 "hello, world"

  • 求余运算符

求余运算符(a % b)是计算 b的多少倍刚刚好可以容入a,返回多出来的那部分(余数)。

【注意】

求余运算符(%)在其他语言也叫取模运算符。但是严格说来,我们看该运算符对负数的操作结果,「求余」「取模」更合适些。

我们来谈谈取余是怎么回事,计算9 % 4,你先计算出 4 的多少倍会刚好可以容入9中:

你可以在 9 中放入两个 4,那余数是 1

在 Swift 中可以表达为:

 9 % 4    // 等于 1

为了得到 a % b 的结果,%计算了以下等式,并输出余数作为结果:

 a = (b × 倍数) + 余数

当倍数取最大值的时候,就会刚好可以容入a中。

94 代入等式中,我们得 1

 9 = (4 × 2) + 1

同样的方法,我们来计算-9 % 4

 -9 % 4   // 等于 -1

-94 代入等式,-2 是取到的最大整数:

 -9 = (4 × -2) + -1

余数是 -1
在对负数b 求余时,b的符号会被忽略。这意味着 a % ba % -b的结果是相同的。


  • 一元负号运算符
    数值的正负号可以使用前缀-(即一元负号符)来切换:

     let three = 3
     let minusThree = -three       // minusThree 等于 -3
     let plusThree = -minusThree   // plusThree 等于 3, 或 "负负3"
    

一元负号符(-)写在操作数之前,中间没有空格。


  • 一元正号运算符
    一元正号符(+)不做任何改变地返回操作数的值:

     let minusSix = -6
     let alsoMinusSix = +minusSix  // alsoMinusSix 等于 -6
    

虽然一元正号符什么都不会改变,但当你在使用一元负号来表达负数时,你可以使用一元正号来表达正数,如此你的代码会具有对称美。

组合赋值运算符
如同 C 语言,Swift 也提供把其他运算符和赋值运算(=)组合的组合赋值运算符,组合加运算(+=)是其中一个例子:

   var a = 1
   a += 2
   // a 现在是 3

表达式 a += 2a = a + 2 的简写,一个组合加运算就是把加法运算和赋值运算组合成进一个运算符里,同时完成两个运算任务。

【注意:】
复合赋值运算没有返回值, let b = a += 2这类代码是错误。这不同于上面提到的自增和自减运算符。
更多 Swift 标准库运算符的信息,请看运算符声明

比较运算符(Comparison Operators)
所有标准 C 语言中的比较运算符都可以在 Swift 中使用:

 - 等于(a == b)
 - 不等于(a != b)
 - 大于(a > b)
 - 小于(a < b)
 - 大于等于(a >= b)
 - 小于等于(a <= b)

注意: Swift 也提供恒等(===)和不恒等(!==)这两个比较符来判断两个对象是否引用同一个对象实例。更多细节在类与结构

每个比较运算都返回了一个标识表达式是否成立的布尔值:

   1 == 1   // true, 因为 1 等于 1
   2 != 1   // true, 因为 2 不等于 1
   2 > 1    // true, 因为 2 大于 1
   1 < 2    // true, 因为 1 小于2
   1 >= 1   // true, 因为 1 大于等于 1
   2 <= 1   // false, 因为 2 并不小于等于 1

比较运算多用于条件语句,如if条件:

 let name = "world"
 if name == "world" {
     print("hello, world")
 } else {
    print("I'm sorry \(name), but I don't recognize you")
 }
 // 输出 "hello, world", 因为 `name` 就是等于 "world"

关于 if 语句,请看控制流

如果两个元组的元素相同,且长度相同的话,元组就可以被比较。比较元组大小会按照从左到右、逐值比较的方式,直到发现有两个值不等时停止。如果所有的值都相等,那么这一对元组我们就称它们是相等的。例如:

 (1, "zebra") < (2, "apple")   // true,因为 1 小于 2
 (3, "apple") < (3, "bird")    // true,因为 3 等于 3,但是 apple 小于 bird
 (4, "dog") == (4, "dog")      // true,因为 4 等于 4,dog 等于 dog

在上面的例子中,你可以看到,在第一行中从左到右的比较行为。因为1小于2,所以(1, "zebra")小于(2, "apple"),不管元组剩下的值如何。所以zebra大于apple对结果没有任何影响,因为元组的比较结果已经被第一个元素决定了。不过,当元组的第一个元素相同时候,第二个元素将会用作比较-第二行和第三行代码就发生了这样的比较。

当元组中的元素都可以被比较时,你也可以使用这些运算符来比较它们的大小。例如,像下面展示的代码,你可以比较两个类型为 (String, Int)的元组,因为IntString类型的值可以比较。相反,Bool不能被比较,也意味着存有布尔类型的元组不能被比较。

 ("blue", -1) < ("purple", 1)       // 正常,比较的结果为 true
 ("blue", false) < ("purple", true) // 错误,因为 < 不能比较布尔类型

【注意】
Swift 标准库只能比较七个以内元素的元组比较函数。如果你的元组元素超过个时,你需要自己实现比较运算符。

三目运算符(Ternary Conditional Operator)
三目运算符的特殊在于它是有三个操作数的运算符,它的形式是 问题 ?答案 1 : 答案2。它简洁地表达根据 问题成立与否作出二选一的操作。如果 问题 成立,返回 答案 1 的结果;反之返回 答案 2 的结果。

三目运算符是以下代码的缩写形式:

 if question {
     answer1
 } else {
     answer2
 }

这里有个计算表格行高的例子。如果有表头,那行高应比内容高度要高出 50 点;如果没有表头,只需高出 20点:

 let contentHeight = 40
 let hasHeader = true
 let rowHeight = contentHeight + (hasHeader ? 50 : 20)
 // rowHeight 现在是 90

上面的写法比下面的代码更简洁:

 let contentHeight = 40
 let hasHeader = true
 var rowHeight = contentHeight
 if hasHeader {
    rowHeight = rowHeight + 50
 } else {
     rowHeight = rowHeight + 20
 }
 // rowHeight 现在是 90

第一段代码例子使用了三目运算,所以一行代码就能让我们得到正确答案。这比第二段代码简洁得多,无需将 rowHeight 定义成变量,因为它的值无需在if语句中改变。

三目运算提供有效率且便捷的方式来表达二选一的选择。需要注意的事,过度使用三目运算符会使简洁的代码变的难懂。我们应避免在一个组合语句中使用多个三目运算符。

空合运算符(Nil Coalescing Operator)

空合运算符(a ?? b)将对可选类型a进行空判断,如果 a 包含一个值就进行解封,否则就返回一个默认值b。表达式 a 必须是 Optional 类型。默认值b的类型必须要和a存储值的类型保持一致。

空合运算符是对以下代码的简短表达方法:

 a != nil ? a! : b

上述代码使用了三目运算符。当可选类型 a的值不为空时,进行强制解封(a!),访问 a 中的值;反之返回默认值b。无疑空合运算符(??)提供了一种更为优雅的方式去封装条件判断和解封两种行为,显得简洁以及更具可读性。

注意: 如果a为非空值(non-nil),那么值b将不会被计算。这也就是所谓的短路求值。

下文例子采用空合运算符,实现了在默认颜色名和可选自定义颜色名之间抉择:

 let defaultColorName = "red"
 var userDefinedColorName: String?   //默认值为 nil

 var colorNameToUse = userDefinedColorName ?? defaultColorName
 // userDefinedColorName 的值为空,所以 colorNameToUse 的值为 "red"

userDefinedColorName 变量被定义为一个可选的String类型,默认值为 nil。由于 userDefinedColorName是一个可选类型,我们可以使用空合运算符去判断其值。在上一个例子中,通过空合运算符为一个名为 colorNameToUse的变量赋予一个字符串类型初始值。 由于 userDefinedColorName值为空,因此表达式userDefinedColorName ?? defaultColorName返回 defaultColorName的值,即 red

另一种情况,分配一个非空值(non-nil)userDefinedColorName,再次执行空合运算,运算结果为封包在userDefaultColorName中的值,而非默认值。

 userDefinedColorName = "green"
 colorNameToUse = userDefinedColorName ?? defaultColorName
 // userDefinedColorName 非空,因此 colorNameToUse 的值为 "green"

区间运算符(Range Operators)
Swift 提供了几种方便表达一个区间的值的区间运算符。


  • 闭区间运算符
    闭区间运算符(a...b)定义一个包含从ab(包括ab)的所有值的区间。a 的值不能超过b。 ‌ 闭区间运算符在迭代一个区间的所有值时是非常有用的,如在 for-in循环中:

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

关于 for-in,请看控制流

半开区间运算符
半开区间运算符(a..<b)定义一个从 ab但不包括b的区间。 之所以称为半开区间,是因为该区间包含第一个值而不包括最后的值。

半开区间的实用性在于当你使用一个从 0开始的列表(如数组)时,非常方便地从0数到列表的长度。

 let names = ["Anna", "Alex", "Brian", "Jack"]
 let count = names.count
 for i in 0..<count {
     print("第 \(i + 1) 个人叫 \(names[i])")
 }
 // 第 1 个人叫 Anna
 // 第 2 个人叫 Alex
 // 第 3 个人叫 Brian
 // 第 4 个人叫 Jack

数组有 4 个元素,但 0..<count 只数到3(最后一个元素的下标),因为它是半开区间。关于数组,请查阅数组

单侧区间
闭区间操作符有另一个表达形式,可以表达往一侧无限延伸的区间 —— 例如,一个包含了数组从索引2 到结尾的所有值的区间。在这些情况下,你可以省略掉区间操作符一侧的值。这种区间叫做单侧区间,因为操作符只有一侧有值。例如:

  for name in names[2...] {
     print(name)
 }
 // Brian
 // Jack

 for name in names[...2] {
     print(name)
 }
 // Anna
 // Alex
 // Brian

半开区间操作符也有单侧表达形式,附带上它的最终值。就像你使用区间去包含一个值,最终值并不会落在区间内。例如:

 for name in names[..<2] {
     print(name)
 }
 // Anna
 // Alex

单侧区间不止可以在下标里使用,也可以在别的情境下使用。你不能遍历省略了初始值的单侧区间,因为遍历的开端并不明显。你可以遍历一个省略最终值的单侧区间;然而,由于这种区间无限延伸的特性,请保证你在循环里有一个结束循环的分支。你也可以查看一个单侧区间是否包含某个特定的值,就像下面展示的那样。

 let range = ...5
 range.contains(7)   // false
 range.contains(4)   // true
 range.contains(-1)  // true

逻辑运算符(Logical Operators)

逻辑运算符的操作对象是逻辑布尔值。Swift 支持基于C 语言的三个标准逻辑运算。

  • 逻辑非(!a)
  • 逻辑与(a && b)
  • 逻辑或(a || b)

逻辑非运算符
逻辑非运算符(!a)对一个布尔值取反,使得 truefalsefalsetrue

它是一个前置运算符,需紧跟在操作数之前,且不加空格。读作 非a ,例子如下:

 let allowedEntry = false
 if !allowedEntry {
     print("ACCESS DENIED")
 }
 // 输出 "ACCESS DENIED"

if !allowedEntry 语句可以读作「如果非allowedEntry」,接下一行代码只有在「非allowedEntry」为 true,即 allowEntryfalse时被执行。

在示例代码中,小心地选择布尔常量或变量有助于代码的可读性,并且避免使用双重逻辑非运算,或混乱的逻辑语句。

逻辑与运算符

逻辑与运算符 (a && b) 表达了只有 ab 的值都为 true 时,整个表达式的值才会是 true

只要任意一个值为 false ,整个表达式的值就为 false 。事实上,如果第一个值为 false ,那么是不去计算第二个值的,因为它已经不可能影响整个表达式的结果了。这被称做短路计算 (short-circuit evaluation)

以下例子,只有两个 Bool 值都为 true 的时候才允许进入 if

 let enteredDoorCode = true
 let passedRetinaScan = false
 if enteredDoorCode && passedRetinaScan {
    print("Welcome!")
 } else {
     print("ACCESS DENIED")
 }
 // 输出 "ACCESS DENIED"

逻辑或运算符

逻辑或运算符 (a || b)是一个由两个连续的 |组成的中置运算符。它表示了两个逻辑表达式的其中一个为 true,整个表达式就为 true

同逻辑与运算符类似,逻辑或也是「短路计算」的,当左端的表达式为true时,将不计算右边的表达式了,因为它不可能改变整个表达式的值了。

以下示例代码中,第一个布尔值(hasDoorKey)false,但第二个值(knowsOverridePassword)true,所以整个表达是true,于是允许进入:

 let hasDoorKey = false
 let knowsOverridePassword = true
 if hasDoorKey || knowsOverridePassword {
     print("Welcome!")
 } else {
     print("ACCESS DENIED")
 }
 // 输出 "Welcome!"

逻辑运算符组合计算

我们可以组合多个逻辑运算符来表达一个复合逻辑:

 if enteredDoorCode && passedRetinaScan || hasDoorKey || >knowsOverridePassword {
   print("Welcome!")
 } else {
     print("ACCESS DENIED")
 }
 // 输出 "Welcome!"

这个例子使用了含多个 &&||的复合逻辑。但无论怎样, &&||始终只能操作两个值。所以这实际是三个简单逻辑连续操作的结果。我们来解读一下:

如果我们输入了正确的密码并通过了视网膜扫描,或者我们有一把有效的钥匙,又或者我们知道紧急情况下重置的密码,我们就能把门打开进入。

前两种情况,我们都不满足,所以前两个简单逻辑的结果是 false,但是我们是知道紧急情况下重置的密码的,所以整个复杂表达式的值还是 true

注意: Swift逻辑操作符 &&||是左结合的,这意味着拥有多元逻辑操作符的复合表达式优先计算最左边的子表达式。

使用括号来明确优先级

为了一个复杂表达式更容易读懂,在合适的地方使用括号来明确优先级是很有效的,虽然它并非必要的。在上个关于门的权限的例子中,我们给第一个部分加个括号,使它看起来逻辑更明确:

 if (enteredDoorCode && passedRetinaScan) || hasDoorKey || >knowsOverridePassword {
     print("Welcome!")
 } else {
     print("ACCESS DENIED")
 }
 // 输出 "Welcome!"

这括号使得前两个值被看成整个逻辑表达中独立的一个部分。虽然有括号和没括号的输出结果是一样的,但对于读代码的人来说有括号的代码更清晰。可读性比简洁性更重要,请在可以让你代码变清晰的地方加个括号吧!

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

推荐阅读更多精彩内容