Swift 循环遍历集合方法总结

2,889 阅读8分钟
原文链接: www.jianshu.com

尊重知识,转发请注明出处:Swift循环遍历集合方法总结


概要

2016年2月份我总结过OC循环遍历,文章在此:iOS开发遍历集合(NSArray,NSDictionary、NSSet)方法总结。时隔一年,随着Swift的逐渐完善,自己使用Swift开发的项目经验和知识逐渐积累,是时候总结一下Swift的循环遍历了。
相信Swift一定会给你一些不一样的东西,甚至是惊喜。


Swit-for.jpeg

第一种方式:for-in循环

OC延续了C语言的for循环,在Swift中被彻底改造,我们无法再使用传统形式的for循环了

遍历数组和字典:

    //遍历数组
    let iosArray = ["L", "O", "V", "E", "I", "O", "S"]

    for index in 0...6 {
        print(iosArray[index])
    }

    for index in 0..<6 {
        print(iosArray[index])
    }

    for element in iosArray {
        print(element)
    }

    //遍历字典
    let iosDict = ["1": "one", "2": "two", "3": "three", "4": "four"]
    for (key, value) in iosDict {
        print("\(key): \(value)")
    }

    //单独遍历字典的key和value
    let keys = iosDict.keys
    for k in keys {
        print(k)
    }

    let values = iosDict.values
    for v in values {
        print(v)
    }

如上遍历数组使用了2种方式

1、
第一种方式是Swift中普通的for循环语法,在索引index和遍历范围0...6之间用关键字in,这里要注意0...6的表示的范围是:0<= index <= 6,而0..<6表示的是:0<= index < 6,这里要注意的是没有:0<..6的形式。只要熟悉了Swift语法,以上这些并不难理解。
拓展1:0...6的形式还可以取出制定范围的数组中的元素,代码如下:

    let sectionArray = iosArray[1...4]
    print(sectionArray)
    输出:
    ▿ 4 elements
        - 0 : "O"
    - 1 : "V"
    - 2 : "E"
    - 3 : "I"

拓展2:0...6的形式还可以用来初始化创建数组,代码如下:

    let numbers = Array(1...7)
    print(numbers)
    输出:
    ▿ 7 elements
    - 0 : 1
    - 1 : 2
    - 2 : 3
    - 3 : 4
    - 4 : 5
    - 5 : 6
    - 6 : 7

也就是说以后遇到涉及范围的情况都可以尝试0...6这种形式,看看是否可以迅速获取指定范围内的元素,可用的地方还有很多,小伙伴自己掘金吧。

2、
第二种方式类似于OC中的快速遍历,不需要索引直接就可以访问到数组中的元素,也很好理解。


字典的遍历可分为同时或者分别遍历key和value

1、
同时遍历key和value时利用了Swift的元组,元组可以把不同类型的值组合成一个复合的值,使用起来非常方便,这样就可以同时拿到字典的key和value了。

2、
单独遍历字典的key个value时,需要注意的是,keys和values并不是Array,因此无法直接使用keys[0]的形式访问,他们实际的类型是LazyMapCollection<[Key : Value], Key>,显然不是一个数组。当然我们可以将他们转换成数组,如下:

    //将字典的kyes转换成数组
    let keys = Array(iosDict.keys)
    print(keys[0])

由于字典是无序的,所有这么做的意义并不大。

第二种方式:Swift为for循环带来的惊喜

将以下内容单拿出来作为第二种方式不太合适,其实这部分还是属于Swift的for-in循环,单独拿出来是出于对这种方式的喜爱,也让大家在看的时候更加醒目。

反向遍历

    //倒序遍历数组
    for index in (0...6).reversed() {
        print(iosArray[index])
    }

    for element in iosArray.reversed() {
        print(element)
    }

    //倒序遍历字典
    for (key, value) in iosDict.reversed() {
        print("\(key): \(value)")
    }

1、
如上无论是0...6这种索引方式还是快速遍历,都可直接调用reversed()函数轻松实现反向遍历。

2、
对于字典的反向遍历,有些小伙伴可能会有些疑问,字典是无序的,反向和正向遍历有区别吗,似乎意义不大。这里需要说明的是,字典的无序是说不保证顺序,但是在内存中是按照顺序排列的,只是这种顺序不一定按照我们存入或者编码的顺序排列,因此字典的反向遍历也是有意义的。

3、
看过我去年总结的OC循环遍历的小伙伴一定还记得,当我们需要在遍历集合时改变集合中的元素时,正向遍历会偶尔出现崩溃的问题,尤其是数据量较大时几乎每次都会崩溃,当我们使用反向遍历时就没有崩溃的问题了,在Swift中为了保证程序的稳定,也建议在遍历集合需要修改集合元素时采用反向遍历。

拓展:reversed()函数实际上是返回给我们一个顺序完全颠倒的集合,那么我们就可以利用这个函数得到一个倒序的集合,非常方便,代码如下:

    //获取倒序数组
    let reversedArray = Array(iosArray.reversed())
    print(reversedArray)

forEach遍历

如果还有小伙伴认为for-in遍历繁琐,Swift还提供了一种更加简洁的遍历方式forEach,代码如下:

    //使用forEach正向遍历
    iosArray.forEach { (word) in
        print(word)
    }

    //使用forEach的反向遍历
    iosArray.reversed().forEach { (word) in
        print(word)
    }

注意: 1、不能使用“break”或者“continue”退出遍历; 2、使用“return”结束当前循环遍历,这种方式只是结束了当前闭包内的循环遍历,并不会跳过后续代码的调用。


stride遍历

stride遍历分为
stride<T : Strideable>(from start: T, to end: T, by stride: T.Stride)

stride<T : Strideable>(from start: T, through end: T, by stride: T.Stride)
两种遍历方式,代码如下:

    //stride正向遍历
    for index in stride(from: 1, to: 6, by: 1) {
        print(index)
        print(iosArray[index])
    }

    //stride正向跳跃遍历
    for index in stride(from: 0, to: 6, by: 2) {
        print(index)
        print(iosArray[index])
    }

    //stride反向遍历
    for index in stride(from: 6, to: 1, by: -1) {
        print(index)
        print(iosArray[index])
    }

    //stride through正向遍历
    for index in stride(from: 0, through: 6, by: 1) {
        print(index)
        print(iosArray[index])
    }

1、
正如stride单词的含义“大步跨过”,使用这种方式遍历的好处自然是可以灵活的根据自己的需求遍历,比如我们有时需要遍历索引为偶数或者基数的元素,或者每隔3个元素遍历一次等等类似的需求都可以轻松实现;

2、
stride遍历同样可以实现正向和反向的遍历,在by后面添加正数表示递增的正向遍历,添加负数表示递减的反向遍历;

3、
to和through两种遍历方式的不同在于to不包含后面的索引,而through包含后面的索引,以to: 6through: 6为例,to:<6或者>6through:<=6或者>=6,至于是<还是>取决于是正向遍历还是反向遍历。

第三种方式:基于块的遍历

OC拥有一套很优雅基于快的遍历,Swift保持了这套优秀的接口,下面来看看Swift是如何使用的。

正向遍历

    //遍历数组
    for (n, c) in iosArray.enumerated() {
        print("\(n): \(c)")
    }

    //遍历字典
    for (n, c) in iosDict.enumerated() {
        print("\(n): \(c)")
    }

注意: 1、(n, c)中n表示元素的输入顺序,c表示集合中的每一个元素; 2、由于数组是有序的,所以在数组中n自然也可以表示每一个元素在数组中索引,而字典是无序的,但是n依然会按照0、1、2...的顺序输入,因此不可以代表在字典中的索引。


反向遍历

    //反向遍历数组
    for (n, c) in iosArray.enumerated().reversed() {
        print("\(n): \(c)")
    }

    //反向遍历字典
    for (n, c) in iosDict.enumerated().reversed() {
        print("\(n): \(c)")
    }

反向遍历就是直接在enumerated()函数后调用reversed()函数。

总结

在总结OC循环遍历时,笔者极力推崇基于块的循环遍历,因为相比较其他的遍历方式,基于块的循环遍历实在是集优雅与实用于一体的尤物,但是在Swift中情况发生了一些变化。

1、
以上列举了3大种遍历方式,其实这种分类方式并不严谨,他们只是展示的形式不一样,本质上都属于for-in循环的变种,都是基于枚举的循环遍历,所以大家不必太较真他们的本质区别,在形式上区分开就可以,不影响我们的使用;

2、
在OC中除了少数情况我们需要使用for (int i = 0; i < n; i++) 的方式,我们都推荐使用基于枚举的快速遍历。在Swift中舍弃了for (int i = 0; i < n; i++)的形式,带给我们for index in 0...6,这并不只是语法格式的变化,从本质上已经完全不一样,使用起来更加方便,也拥有更多的接口提供便利的功能。并且相比较其他的遍历方式也有很多优点,因此在Swift中我们无法一边倒的选择一种方式,应该根据情况选择合适的方法。

3、
在OC中基于块的遍历还有一种情况是并发遍历,我在Swift中没有找到相应的方法,看了几千行代码也没有发现踪迹,有找到的小伙伴还请告知,感激不尽。当然并发遍历用的并不多,我们很多时候我们都希望集合的元素按序出现,在时间和效率上也没有区别。

尾巴

抛开我们的使用习惯,Swift在很多时候都要比OC甚至其他编程语言更加简洁实用,虽然现在Swift编程很多还是依赖于OC,甚至Swift这种编译时的语言的动态性依然是基于OC的运行时,还有大量的类似于UIKit的库也都是基于OC。Swift集其他语言优点于一身又不失个性,从for循环等一系列小的地方的改变就可以看出其独特之处。完全取代OC还需时日,可此时投身Swift确是最好时机,你还在等什么?