Swift之集合类型 (Collection Types)(集合篇)

集合(Sets)

集合(Set)用来存储相同类型并且没有确定顺序的值。当 合元素顺序不重要时或者希望确保每个元素只出现一次 时可以使用 合而不是数组。
注意:Swift的 Set 类型被桥接到 Foundation 中的 NSSet 类。

集合类型的哈希值
一个类型为了存储在 集合中,该类型必须是可哈希化的--也就是说,该类型必须提供一个方法来计算它的哈希 值。一个哈希值是 Int 类型的,相等的对象哈希值必须相同,比如 a==b,因此必须 a.hashValue == b.hashValu e

Swift的所有基本类型(比如 String,Int , DoubleBool)默认都是可哈希化的,可以作为 合的值的类型或 者字典的键的类型。没有关联值的枚举成员值(在枚举有讲述)默认也是可哈希化的。

合类型语法

Swift 中的 Set 类型被写为Set<Element> ,这里的 Element 表示 Set中允许存储的类型,和数组不同的是, 合没有等价的简化形式。

创建和构造一个空的 集合

var letters = Set<Character>()
print("letters is of type Set<Character> with \(letters.count) items.") // 打印 "letters is of type Set<Character> with 0 items."

注意:通过构造器,这里的 letters 变量的类型被推断为 Set<Character> 。

此外,如果上下文提供了类型信息,比如作为函数的参数或者已知类型的变量或常量,我们可以通过一个空的数 组字面量创建一个空的 Set :

letters.insert("a")
// letters 现在含有1个 Character 类型的值
letters = []
// letters 现在是一个空的 Set, 但是它依然是 Set<Character> 类型

用数组字面量创建集合

你可以使用数组字面量来构造 合,并且可以使用简化形式写一个或者多个值作为 合元素。
下面的例子创建一个称之为favoriteGenres 的 合来存储 String类型的值:

var favoriteGenres: Set<String> = ["Rock", "Classical", "Hip hop"] 
// favoriteGenres 被构造成含有三个初始值的 集合

这个favoriteGenres变量被声明为“一个 String值的 合”,写为 Set<String>。由于这个特定的 合含有指 定 String 类型的值,所以它只允许存储 String类型值。这里的 favoriteGenres变量有三个 String 类型的初始 值("Rock""Classical""Hip hop" ),并以数组字面量的方式出现。

注意: favoriteGenres 被声明为一个变量(拥有 var 标示符)而不是一个常量(拥有 let 标示符),因为它里面的元素将 会在下面的例子中被增加或者移除。

一个 Set 类型不能从数组字面量中被单独推断出来,因此Set 类型必须显式声明。然而,由于 Swift 的类型推 断功能,如果你想使用一个数组字面量构造一个Set 并且该数组字面量中的所有元素类型相同,那么你无须写出
Set 的具体类型。 favoriteGenres 的构造形式可以采用简化的方式代替:

var favoriteGenres: Set = ["Rock", "Classical", "Hip hop"]

由于数组字面量中的所有元素类型相同,Swift可以推断出 Set<String> 作为 favoriteGenres变量的正确类型。

访问和修改一个集合
你可以通过 Set 的属性和方法来访问和修改一个 Set

为了找出一个Set 中元素的数量,可以使用其只读属性count :

print("I have \(favoriteGenres.count) favorite music genres.") // 打印 "I have 3 favorite music genres.

使用布尔属性 isEmpty作为一个缩写形式去检查 count属性是否为 0:

 if favoriteGenres.isEmpty {
     print("As far as music goes, I'm not picky.")
 } else {
     print("I have particular music preferences.")
}
// 打印 "I have particular music preferences."

你可以通过调用 Setinsert(_:)方法来添加一个新元素:

favoriteGenres.insert("Jazz")
// favoriteGenres 现在包含4个元素

你可以通过调用 Setremove(_:)方法去删除一个元素,如果该值是该 Set的一个元素则删除该元素并且返回 被删除的元素值,否则如果该Set 不包含该值,则返回 nil 。另外, Set 中的所有元素可以通过它的removeAl l()方法删除。

 if let removedGenre = favoriteGenres.remove("Rock") {
     print("\(removedGenre)? I'm over it.")
 } else {
     print("I never much cared for that.")
}
// 打印 "Rock? I'm over it."

使用contains(_:) 方法去检查Set 中是否包含一个特定的值:

遍历一个集合

你可以在一个for-in循环中遍历一个 Set中的所有值。

 for genre in favoriteGenres {
     print("\(genre)")
 }
 // Classical
 // Jazz
 // Hip hop

SwiftSet类型没有确定的顺序,为了按照特定顺序来遍历一个 Set 中的值可以使用 sorted() 方法,它将返
回一个有序数组,这个数组的元素排列顺序由操作符'<'对元素进行比较的结果来确定.

 for genre in favoriteGenres.sorted() {
     print("(genre)")
 }
 // prints "Classical"
 // prints "Hip hop"
 // prints "Jazz

集合操作

你可以高效地完成 Set 的一些基本操作,比如把两个 集合组合到一起,判断两个 合共有元素,或者判断两个 集合是否全包含,部分包含或者不相交。

基本集合操作
• 使用 intersection(:) 方法根据两个 合中都包含的值创建的一个新的 集合。
• 使用 symmetricDifference(
:) 方法根据在一个 集合中但不在两个 合中的值创建一个新的 集合。 • 使用 union(:) 方法根据两个集合的值创建一个新的集 合。
• 使用 subtracting(
:) 方法根据不在该 合中的值创建一个新的 集合。

 let oddDigits: Set = [1, 3, 5, 7, 9]
 let evenDigits: Set = [0, 2, 4, 6, 8]
 let singleDigitPrimeNumbers: Set = [2, 3, 5, 7]
 oddDigits.union(evenDigits).sort()
 // [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
 oddDigits. intersection(evenDigits).sorted()
 // []
 oddDigits.subtracting(singleDigitPrimeNumbers).sorted()
 // [1, 9]  以 singleDigitPrimeNumbers集合为标准,取出oddDigits集合中不在singleDigitPrimeNumbers的值
 oddDigits. symmetricDifference(singleDigitPrimeNumbers).sorted()
 // [1, 2, 9] 取出两个集合的并集去掉交集的集合

集合成员关系和相等

• 使用“是否相等”运算符( == )来判断两个 集合是否包含全部相同的值。
• 使用 isSubset(of:) 方法来判断一个集 合中的值是否也被包含在另外一个 合中。
• 使用 isSuperset(of:) 方法来判断一个 集合中包含另一个集合中所有的值。
• 使用 isStrictSubset(of:) 或者 isStrictSuperset(of:) 方法来判断一个 合是否是另外一个集 合的子集 合或 者父集 合并且两个集 合并不相等。
• 使用 isDisjoint(with:) 方法来判断两个集 合是否不含有相同的值(是否没有交集)。

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

推荐阅读更多精彩内容