Swift支持C標准庫中的大多數運算符並提升了各自的兼容性,從而可以排除常見的編碼錯誤。賦值操作符 (=)不會返回一個值,這樣可以防止你因粗心將賦值運算符 (=)寫成 (==)而引起錯誤。算術符(+、 -、 *、 /、 % 等)會檢查與駁回值溢出,這樣可以避免值類型的數據在超過值類型所允許的存儲范圍時,出現意想不到的數據。你可以選擇使用Swift所提供的值溢出運算符進行量化溢出的行為,詳細見溢出操作符。
與C語言不同,Swift允許你對浮點數執行取余運算。同時,Swift提供兩個范圍的運算符 (a..b 和 a…b),作為表示一個數值范圍的簡寫方式,這點C不支持。
本章節描述了Swift常見運算符。高級運算符覆蓋了Swift的高級操作符,並且對自定義操作符,對自定義類型操作符的實現進行了描述。
操作符都是一元、二元或三元:
操作符所影響的值被稱為操作數。表達式1 + 2中,符號 + 是一個二元運算符並且兩個操作數分別為 1 和 2。
賦值運算符(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支持所有數字類型的四個標注運算符:
例如:
1 + 2 // equals 3
5 - 3 // equals 2
2 * 3 // equals 6
10.0 / 2.5 // equals 4.0
不同於C和Objective-C,默認情況下Swift的算術運算符不允許值溢出。你可以通過Swift的溢出運算符來選擇值的溢出情況(例如 a & + b)。詳見Overflow Operators
加法運算符對字符串連接也一樣適用,例如:
"hello, " + "world" // equals "hello, world"
兩個字符,或者一個字符一個字符串,能組合成一個新的字符串:
let dog: Character = "dog"(由於浏覽器不能顯示狗的unicode圖像,故用三個字母代替……)
let cow: Character = "cow"(同上……)
let dogCow = dog + cow
// dogCow is equal to "dogcow"
詳見Concatenating Strings and Characters
取余運算符(a % b)計算出a是b的幾倍然後返回被留下的值(余數)。
注:余數運算符(%)亦稱是其他語言的一個取模運算符。然而,其在Swift裡意味著如果對負數操作,嚴格上講,得到的是余數而不是模數。
這是余數運算符如何工作。 要計算9% 4,你首先得求出9是4的幾倍 :
9能去除兩個4,並且余數是1 (顯示在橙色)。
在Swift中,這個將被寫成:
9 % 4 // equals
確定a % b的答案, 運算符%計算下列等式並且返回余數作為其輸出:
a = (b × some multiplier) + remainder
some multiplier 是a裡面能包含b的最多倍數。
將9和4插入到公式:
9 = (4 × 2) + 1
同一個方法是應用的,當計算a時的一個負值的余數:
-9 % 4 // equals -1
將-9和4插入到公式:
-9 = (4 × -2) + -1
產生余數值為-1。
b為負值時的b的符號被忽略,這意味著%b和%-b的結果是一樣的。、
不同於C和Objective-C,Swift的余數運算符也能運用於浮點數:
8 % 2.5 // equals 0.5
在本例中, 8用2.5來分等於3, 余數是0.5,因此余數為0.5。
像C一樣,Swift提供一個自增運算符(++)和自減運算符(–)作為增加或減少一個數值的一種快捷方式,增減量為1。 您能對任何整數或浮點類型的變量使用這些運算符。
var i = 0
++i // i now equals 1
每當你使用 ++i ,i 的值增加1,本質上++i可以看做是i=i+1,同樣–i可以看做是i=i-1。
++和–符號可以使用作為前綴算符或作為後綴運算符。++i 和 i++ 是兩個有效的方式給i的值增加1,同樣, –i和i–如是。
注意這些運算符修改i並且返回值。如果你只想要增加或減值i,您可以忽略返回值。然而,如果你使用返回值,根據下列規則將是不同的根據的您是否使用了運算符的前綴或後綴版本,它:
例如:
var a = 0
let b = ++a
// a and b are now both equal to 1
let c = a++
// a is now equal to 2, but c has been set to the pre-increment value of 1
在上面的例子中,let b = ++a 中a在返回其值之前增加,這就是為什麼a和b的新值是等於1。
然而,let c = a++ 中a在返回其值之後增加,這意味著c獲得a的原值1,然後a自增,a等於2。
除非你需要特定工作情況下才使用i++,否則在所有的情況下建議你使用++i和–i, 因為他們修改i並返回值的行為符合我們的預期。
一個數值前加了符號-,叫作一元減運算符:
let three = 3
let minusThree = -three // minusThree equals -3
let plusThree = -minusThree // plusThree equals 3, or "minus minus three"
一元減運算符(-)直接地被加在前面,在它起作用的值之前,不用任何空白空間。
一元加運算符(+)返回它起作用的值,不做任何變動:
let minusSix = -6
let alsoMinusSix = +minusSix // alsoMinusSix equals -6
雖然一元加上運算符實際上不執行什麼,當你也使用一元減負數的運算符時,你能使用它提供對稱的正數。
Swift提供類似C語言的復合賦值操作符,即把賦值和另一個運算合並起來。舉個例子,像加法賦值運算符(+ =):
var a = 1
a += 2
// a is now equal to 3
表達式 a += 2 比 a = a + 2更精煉。加法賦值運算符能夠有效地把加法和賦值組合到一個運算,同時執行這兩個任務。
要注意的是,復合賦值操作符不返回值。例如,你不能寫讓成let b = + = 2,這種行為不同於上面提到的遞增和遞減運算符。
復合賦值運算符的完整列表可以在[Expressions]那一章節找到
Swift支持所有標准c的比較運算符
注:Swift 提供兩個恆等運算符(=== and !==),用它來測試兩個對象引用是否來自於同一個對象實例。詳見Classes and Structures。 每個比較操作符返回一個Bool值來表示語句是否為真:
1 == 1 // true, because 1 is equal to 1
2 != 1 // true, because 2 is not equal to 1
2 > 1 // true, because 2 is greater than 1
1 < 2 // true, because 1 is less than 2
1 >= 1 // true, because 1 is greater than or equal to 1
2 <= 1 // false, because 2 is not less than or equal to 1
比較操作符通常用在條件語句,如if語句:
let name = "world"
if name == "world" {
println("hello, world")
} else {
println("I'm sorry \(name), but I don't recognize you")
}
// prints "hello, world", because name is indeed equal to "world"
想要了解更多有關的if語句,請參閱控制流。
三元條件運算符是一種特殊的運算符,有三個部分,其形式為question? answer1:answer2.這是一個用來測試兩種表達式基於輸入是真或是 假的快捷方式。如果question? 為真時, 它評估answer1並返回其值; 否則,它評估answer2並返回其值。三元條件運算符是下面的代碼的簡化:
if question {
answer1
} else {
answer2
}
這裡舉一個列子,計算一個表行像素的高度,如果行有一個頭,行高應該是50像素,比內容要高度要高。如果行沒有頭是20像素:
let contentHeight = 40
let hasHeader = true
let rowHeight = contentHeight + (hasHeader ? 50 : 20)
// rowHeight is equal to 90
前面的例子也可以用下面的的代碼:
let contentHeight = 40
let hasHeader = true
var rowHeight = contentHeight
if hasHeader {
rowHeight = rowHeight + 50
} else {
rowHeight = rowHeight + 20
}
// rowHeight is equal to 90
第一個例子使用的三元條件運算符,意味著rowHeight可以在一行代碼被設置為正確的值。這比第二個示例更簡潔, 不需要課外的rowHeight變量, 因為它的價值不需要在一個if語句中修改。
三元條件運算符提供了一個高效的寫法來決定哪個表達式會被執行。不過還是請小心使用三元條件運算符,其簡潔性如果過度使用會導致閱讀代碼的困難。要避免多個實例的三元條件運算符組合成一個復合語句。
Swift包含兩個范圍運算符,能快捷的表達一系列的值
封閉范圍運算符(a…b)定義了一個范圍,從a到b,並包括a和b的值。
當要在一個范圍內迭代所有可能的值的時候,范圍運算符是非常有用的, 例如for-in循環
for index in 1...5 {
println("\(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
欲了解更多for-in循環,請參閱控制流。
半封閉的區域運算符(a..b)定義了從a到b的范圍,但不包括b。它被認為是半封閉的,因為它包含第一個值,而不包含最終值。
半封閉的范圍使用明確,當你使用從零開始的列表,如數組,它是有用的數到(但不包括)列表的長度:
let names = ["Anna", "Alex", "Brian", "Jack"]
let count = names.count
for i in 0..count {
println("Person \(i + 1) is called \(names[i])")
}
// Person 1 is called Anna
// Person 2 is called Alex
// Person 3 is called Brian
// Person 4 is called Jack
請注意,該數組包含四個項目,但0 . .數只數到3(數組中的最後一個項目的索引),因為它是一個半封閉的范圍。欲了解更多有關數組的信息,請參閱數組
邏輯運算符修改或結合布爾邏輯值true和false。Swift支持這三個標准邏輯運算符基於c語言:
邏輯非運算符(!a)轉化一個Bollean值,true變成false,false變成true。
邏輯操作符是一個前綴操作符, 並立即出現在它修飾的值之前,沒有任何空白,它被解讀為”不是”, 見下面的例子:
let allowedEntry = false
if !allowedEntry {
println("ACCESS DENIED")
}
// prints "ACCESS DENIED"
這句話if !allowedEntry 能理解為 “if not allowedEntry.” 只執行後續的行,如果“not allowedEntry” 是 true; 那就是說 if allowedEntry是false.
在這個例子中,精心挑選的布爾常量和變量名可以幫助保持代碼的可讀性和簡潔,同時避免雙重否定或混亂的邏輯語句。
邏輯與運算符:(A && B)創建的表達式中,A和B兩個值必須同時為true時表達式才正確。
其中A或者B有任一值是false時,邏輯與算符表示不成立,必須兩者同時為true時才成立。事實上,如果第一個值是false,第二個值甚至不會再進行判斷,因為必須是兩個值皆為true,已經有一方false、則沒必要再往下面進行判斷了。這被稱作短路條件。
以下這個例子判斷兩個Bool 類型的值,並只有這兩個值都為真的時候會輸出:Welcome。失敗則輸出”ACCESS DENIED”:
let enteredDoorCode = true
let passedRetinaScan = false
if enteredDoorCode && passedRetinaScan {
println("Welcome!")
} else {
println("ACCESS DENIED")
}
// prints "ACCESS DENIED”
表達式(a || b)運算符中、只要a或者b有一個為true,表達式就成立。
與上面的邏輯與運算符相似,邏輯或運算符使用短路條件判斷,如果左邊是 true,那麼右邊不會被判斷,因為整體結果不會改變了。
在下面的例子中,第一個布爾值(hasDoorKey)為false,但第二個值(knowsOverridePassword)為true。因為兩者有一個值是true,整個表達式的計算結果也為true,正確輸出:Welcome!
let hasDoorKey = false
let knowsOverridePassword = true
if hasDoorKey || knowsOverridePassword {
println("Welcome!")
} else {
println("ACCESS DENIED")
}
// prints "Welcome!"
你可以將多個邏輯運算符復合來創建更長的復合表達式:
if enteredDoorCode && passedRetinaScan || hasDoorKey || knowsOverridePassword {
println("Welcome!")
} else {
println("ACCESS DENIED")
}
// prints "Welcome!"
相比於之前兩個單獨分開的運算符,本次通過多重嵌套、將我們上面的&&、|| 運算符相結合組合成一個較長的復合表達式。看起來有點饒人、其實本質還是兩兩相比較、可以簡單地看成A && B || C || D、從左往右根據運算符優先級進行判斷、注意區分開&&、||、只要牢記運算邏輯&&需要兩者都為true、||則只需要一方為true則運算符正確即可解析整個復合表達式、透過現象看本質。
復合表達式中,我們可以添加進()使確邏輯意圖更加明確,上面的例子中,我們可以在第一部分上加括號來使意義更明確。
if (enteredDoorCode && passedRetinaScan) || hasDoorKey || knowsOverridePassword {
println("Welcome!")
} else {
println("ACCESS DENIED")
}
// prints "Welcome!"