1、Range
我們首先創(chuàng)建一個(gè)Range對象。
var x = 1...10
變量x表示了一個(gè)范圍,范圍的起始點(diǎn)為1,終點(diǎn)為10,并且包括1和10。由于Range實(shí)現(xiàn)了CollectionType協(xié)議,所以,如下所示,我們完全可以把Range視為一個(gè)集合。
x.count // 10 集合中包含的元素個(gè)數(shù)
x.contains(5) // ture 集合中包含元素5
x.contains(13) // false
正是由于Range實(shí)現(xiàn)了集合協(xié)議,所以它可以用于for循環(huán)結(jié)構(gòu)。
for value in x {
print(value) // 輸出 1,2,3,4,5,6,7,8,9,10
}
官方文檔把Range成為索引集合,它擅長和集合類型配合來生成切片。例如,我們可以用range方便的在數(shù)組上做一個(gè)切片。
var array = [1,2,3,4,5,6,7,8,9]
array[2...5] // [3,4,5,6] 用range在數(shù)據(jù)上面做切片
注意 數(shù)組切片的類型為 ArraySlice<Int> ,切片后獲得的只是原數(shù)組array的一個(gè)視圖(與原數(shù)組共享同一段內(nèi)存空間),并沒有分配新的內(nèi)存空間。
Range的定義還有其它的表達(dá)方式
var y = 1..<10
這種寫法很“象形”,我們很容猜到這個(gè)寫法的含義。1...10表達(dá)的是一個(gè)閉區(qū)間(包括1和10),1..<10表達(dá)的是一個(gè)半開區(qū)間(包括1但不包括10)。從集合的角度它等價(jià)于1...9。swift3把Range的定義完整化了,進(jìn)一步提供另外兩種表達(dá)式。
var z = 1<.<10 // 開區(qū)間,不包括1也不包括10
var u = 1<..10 // 半開區(qū)間,不包括1,但包括10
2、 IntervalType
swift3以后,我們將不再需要IntervalType了,swift將用Range的功能同時(shí)覆蓋Range與Interval兩種類型。當(dāng)然這并不會(huì)影響我們學(xué)習(xí)Interval了。
Range目的用來表達(dá)一個(gè)集合(索引集合),而Interval重點(diǎn)描述一個(gè)區(qū)間,目的是為了實(shí)現(xiàn)模式匹配。我們可以舉個(gè)例子看一下,比如我們想判斷一個(gè)數(shù)字是否在一個(gè)Range內(nèi),我們可以這樣來實(shí)現(xiàn)。
var range = Int.min...0
if range.contains(1) {
print("yes")
}
這個(gè)操作將會(huì)非常耗時(shí),由于range是一個(gè)集合,所以這個(gè)比較的過程是一個(gè)集合元素逐一遍歷的過程,實(shí)際上我們并不希望用這樣的比較方式,我們更傾的應(yīng)該是這樣的比較。
var number = 1
if 1 >= Int.min && 1 <= 0
print("yes")
}
這樣的比較過程相比于之前的遍歷過程,那可謂“不知道高到哪里去了!”,而這就是Interval類型的存在的意義。Interval的設(shè)計(jì)目的是用于模式匹配。之前判斷1是否在Int.min...0區(qū)間內(nèi),我們可以這樣來實(shí)現(xiàn)。
var inteval = Int.min...0
range ~= 1 // false 模式匹配失敗
操作符 ~= 我們成為模式匹配操作符,用于進(jìn)行模式匹配,對于Interval而言這種模式匹配是通過“比較”來完成?,F(xiàn)在我們可以完全理解Range與Interval的區(qū)別和聯(lián)系了,當(dāng)然從今往后它們就是一家啦。
Interval常常被用于switch case分支結(jié)構(gòu)中,可以看另外一篇文章來了解switch中使用Intverval