Swift 语言提供经典的数组和字典两种集合类型来存储集合数据。数组用来按顺序存储同样类型的数据。字典尽管无序存储同样类型数据值可是须要由独有的标识符引用和寻址(就是键值对)。
Swift 语言里的数组和字典中存储的数据值类型必须明白。 这意味着我们不能把不对的数据类型插入当中。 同一时候这也说明我们全然能够对获取出的值类型很自信。 Swift 对显式类型集合的使用确保了我们的代码对工作所须要的类型很清楚,也让我们在开发中能够早早地找到不论什么的类型不匹配错误。
注意:
Swift 的数组结构在被声明成常量和变量或者被传入函数与方法中时会相对于其它类型展现出不同的特性。获取很多其它信息请參见集合的可变性与集合在赋值和复制中的行为章节。
数组
数组使用有序列表存储同样类型的多重数据。同样的值能够多次出如今一个数组的不同位置中。
Swift 数组对存储数据有详细要求。 不同于 Objective-C 的NSArray和NSMutableArray类,他们能够存储不论什么类型的实例并且不提供他们返回对象的不论什么本质信息。在 Swift 中,数据值在被存储进入某个数组之前类型必须明白,方法是通过显式的类型标注或类型判断,并且不是必须是class类型。比如:假设我们创建了一个Int值类型的数组,我们不能往当中插入不论什么不是Int类型的数据。 Swift 中的数组是类型安全的,并且它们中包括的类型必须明白。
数组的简单语法
写 Swift 数组应该遵循像Array<SomeType>这种形式,当中SomeType是这个数组中唯一同意存在的数据类型。我们也能够使用像SomeType[]这种简单语法。虽然两种形式在功能上是一样的,可是推荐较短的那种,并且在本文中都会使用这种形式来使用数组。
数组构造语句
我们能够使用字面语句来进行数组构造,这是一种用一个或者多个数值构造数组的简单方法。字面语句是一系列由逗号切割并由方括号包括的数值。 [value 1, value 2, value 3]。
以下这个样例创建了一个叫做shoppingList而且存储字符串的数组:
var shoppingList: String[] =["Eggs", "Milk"] // shoppingList 已经被构造而且拥有两个初始项。
shoppingList变量被声明为“字符串值类型的数组“,记作String[]。 由于这个数组被规定仅仅有String一种数据结构,所以仅仅有String类型能够在当中被存取。 在这里,shoppinglist数组由两个String值("Eggs"和"Milk")构造,而且由字面语句定义。
注意:
Shoppinglist数组被声明为变量(varkeyword创建)而不是常量(let创建)是由于以后可能会有很多其它的数据项被插入当中。
在这个样例中,字面语句只包括两个String值。匹配了该数组的变量声明(只能包括String的数组),所以这个字面语句的分配过程就是同意用两个初始项来构造shoppinglist。
因为 Swift 的类型判断机制,当我们用字面语句构造仅仅拥有同样类型值数组的时候,我们不必把数组的类型定义清楚。 shoppinglist的构造也能够这样写:
var shoppingList = ["Eggs","Milk"]
由于全部字面语句中的值都是同样的类型,Swift 能够判断出String[]是shoppinglist中变量的正确类型。
訪问和改动数组
我们能够通过数组的方法和属性来訪问和改动数组,或者下标语法。还能够使用数组的仅仅读属性count来获取数组中的数据项数量。
println("The shopping list contains(shoppingList.count) items.") // 输出"The shoppinglist contains 2 items."(这个数组有2个项)
使用布尔项isEmpty来作为检查count属性的值是否为 0 的捷径。
if shoppingList.isEmpty { println("The shopping list is empty.") } else { println("The shopping list is not empty.") } // 打印 "The shoppinglist is not empty."(shoppinglist不是空的)
也能够使用append方法在数组后面加入新的数据项:
shoppingList.append("Flour") // shoppingList 如今有3个数据项,有人在摊煎饼
除此之外,使用加法赋值运算符(+=)也能够直接在数组后面加入数据项:
shoppingList += "Baking Powder" // shoppingList 如今有四项了
我们也能够使用加法赋值运算符(+=)直接加入拥有同样类型数据的数组。
shoppingList += ["ChocolateSpread", "Cheese", "Butter"] // shoppingList 如今有7项了
能够直接使用下标语法来获取数组中的数据项,把我们须要的数据项的索引值放在直接放在数组名称的方括号里:
var firstItem = shoppingList[0] // 第一项是 "Eggs"
注意第一项在数组中的索引值是0而不是1。 Swift 中的数组索引总是从零開始。
我们也能够用下标来改变某个已有索引值相应的数据值:
shoppingList[0] = "Six eggs" // 当中的第一项如今是 "Sixeggs" 而不是 "Eggs"
还能够利用下标来一次改变一系列数据值,即使新数据和原有数据的数量是不一样的。以下的样例把"Chocolate Spread","Cheese",和"Butter"替换为"Bananas"和"Apples":
shoppingList[4...6] = ["Bananas","Apples"] // shoppingList 如今有六项
注意:
我们不能使用下标语法在数组尾部加入新项。假设我们试着用这样的方法对索引越界的数据进行检索或者设置新值的操作,我们会引发一个执行期错误。我们能够使用索引值和数组的count属性进行比較来在使用某个索引之前先检验是否有效。除了当count等于 0 时(说明这是个空数组),最大索引值一直是count - 1,由于数组都是零起索引。
调用数组的insert(atIndex:)方法来在某个详细索引值之前加入数据项:
shoppingList.insert("MapleSyrup", atIndex: 0) // shoppingList 如今有7项 // "Maple Syrup" 如今是这个列表中的第一项
这次insert函数调用把值为"Maple Syrup"的新数据项插入列表的最開始位置,而且使用0作为索引值。
类似的我们能够使用removeAtIndex方法来移除数组中的某一项。这种方法把数组在特定索引值中存储的数据项移除而且返回这个被移除的数据项(我们不须要的时候就能够无视它):
let mapleSyrup =shoppingList.removeAtIndex(0) //索引值为0的数据项被移除 // shoppingList 如今仅仅有6项,并且不包含MapleSyrup // mapleSyrup常量的值等于被移除数据项的值"Maple Syrup"
数据项被移除后数组中的空出项会被自己主动填补,所以如今索引值为0的数据项的值再次等于"Six eggs":
firstItem = shoppingList[0] // firstItem 如今等于 "Sixeggs"
假设我们仅仅想把数组中的最后一项移除,能够使用removeLast方法而不是removeAtIndex方法来避免我们须要获取数组的count属性。就像后者一样,前者也会返回被移除的数据项:
let apples = shoppingList.removeLast() // 数组的最后一项被移除了 // shoppingList如今仅仅有5项,不包含cheese // apples 常量的值如今等于"Apples" 字符串
数组的遍历
我们能够使用for-in循环来遍历全部数组中的数据项:
for item in shoppingList { println(item) } // Six eggs // Milk // Flour // Baking Powder // Bananas
假设我们同一时候须要每个数据项的值和索引值,能够使用全局enumerate函数来进行数组遍历。enumerate返回一个由每个数据项索引值和数据值组成的键值对组。我们能够把这个键值对组分解成暂时常量或者变量来进行遍历:
for (index, value) inenumerate(shoppingList) { println("Item (index + 1): (value)") } // Item 1: Six eggs // Item 2: Milk // Item 3: Flour // Item 4: Baking Powder // Item 5: Bananas
很多其它关于for-in循环的介绍请參见for 循环。
创建而且构造一个数组
我们能够使用构造语法来创建一个由特定数据类型构成的空数组:
var someInts = Int[]() println("someInts is of type Int[]with (someInts。count) items。") // 打印 "someInts isof type Int[] with 0 items。"(someInts是0数据项的Int[]数组)
注意someInts被设置为一个Int[]构造函数的输出所以它的变量类型被定义为Int[]。
除此之外,假设代码上下文中提供了类型信息,比如一个函数參数或者一个已经定义好类型的常量或者变量,我们能够使用空数组语句创建一个空数组,它的写法非常easy:[](一对空方括号):
someInts.append(3) // someInts 如今包括一个INT值 someInts = [] // someInts 如今是空数组,可是仍然是Int[]类型的。
Swift 中的Array类型还提供一个能够创建特定大小而且全部数据都被默认的构造方法。我们能够把准备增加新数组的数据项数量(count)和适当类型的初始值(repeatedValue)传入数组构造函数:
var threeDoubles = Double[](count: 3,repeatedValue:0.0) // threeDoubles 是一种 Double[]数组, 等于 [0.0, 0.0,0.0]
由于类型判断的存在,我们使用这样的构造方法的时候不须要特别指定数组中存储的数据类型,由于类型能够从默认值判断出来:
var anotherThreeDoubles = Array(count: 3,repeatedValue: 2.5) // anotherThreeDoubles is inferred asDouble[], and equals [2.5, 2.5, 2.5]
最后,我们能够使用加法操作符(+)来组合两种已存在的同样类型数组。新数组的数据类型会被从两个数组的数据类型中判断出来:
var sixDoubles = threeDoubles +anotherThreeDoubles // sixDoubles 被判断为Double[], 等于 [0.0, 0.0, 0.0, 2.5, 2.5, 2.5]