如何在Swift中使用索引和元素迭代循环


784

有没有可以用来遍历数组并具有索引和元素的函数,例如Python的枚举?

for index, element in enumerate(list):
    ...

Answers:


1659

是。从Swift 3.0开始,如果需要每个元素的索引及其值,则可以使用该enumerated()方法遍历数组。它返回由索引和数组中每个项目的值组成的对对的序列。例如:

for (index, element) in list.enumerated() {
  print("Item \(index): \(element)")
}

在Swift 3.0之前和Swift 2.0之后,该函数称为enumerate()

for (index, element) in list.enumerate() {
    print("Item \(index): \(element)")
}

在Swift 2.0之前,它enumerate是一个全局函数。

for (index, element) in enumerate(list) {
    println("Item \(index): \(element)")
}

3
尽管它看起来像一个元组,但是在Swift 1.2(不确定2.0)中,enumerate返回一个EnumerateSequence <base:SequenceType>结构。
nstein 2015年

2
@Leviathlon显着或可衡量的性能开销会很重要吗?号
Dan Rosenstark '16

是访问索引的唯一好处是enumerate
亲爱的2016年

29
也许他们会变成enumeratingSwift 4的Gerund。令人兴奋!
Dan Rosenstark '17

3
@Honey for (index, element) in使用时enumerated会产生误导。应该是for (offset, element) in
Leo Dabus

115

迅速5提供了一个名为方法enumerated()Arrayenumerated()具有以下声明:

func enumerated() -> EnumeratedSequence<Array<Element>>

返回一个成对的序列(n,x),其中n代表一个从零开始的连续整数,x代表该序列的一个元素。


在最简单的情况下,可以enumerated()与for循环一起使用。例如:

let list = ["Car", "Bike", "Plane", "Boat"]
for (index, element) in list.enumerated() {
    print(index, ":", element)
}

/*
prints:
0 : Car
1 : Bike
2 : Plane
3 : Boat
*/

但是请注意,您不限于enumerated()与for循环一起使用。实际上,如果您打算将enumerated()for循环用于类似于以下代码的内容,那么您做错了:

let list = [Int](1...5)
var arrayOfTuples = [(Int, Int)]()

for (index, element) in list.enumerated() {
    arrayOfTuples += [(index, element)]
}

print(arrayOfTuples) // prints [(0, 1), (1, 2), (2, 3), (3, 4), (4, 5)]

一个更快的方法是:

let list = [Int](1...5)
let arrayOfTuples = Array(list.enumerated())
print(arrayOfTuples) // prints [(offset: 0, element: 1), (offset: 1, element: 2), (offset: 2, element: 3), (offset: 3, element: 4), (offset: 4, element: 5)]

作为替代方案,你也可以使用enumerated()具有map

let list = [Int](1...5)
let arrayOfDictionaries = list.enumerated().map { (a, b) in return [a : b] }
print(arrayOfDictionaries) // prints [[0: 1], [1: 2], [2: 3], [3: 4], [4: 5]]

此外,虽然有一定的局限性forEach可以是一个很好的替代,以一个for循环:

let list = [Int](1...5)
list.reversed().enumerated().forEach { print($0, ":", $1) }

/*
prints:
0 : 5
1 : 4
2 : 3
3 : 2
4 : 1
*/

通过使用enumerated()makeIterator(),您甚至可以在自己的上手动进行迭代Array。例如:

import UIKit
import PlaygroundSupport

class ViewController: UIViewController {

    var generator = ["Car", "Bike", "Plane", "Boat"].enumerated().makeIterator()

    override func viewDidLoad() {
        super.viewDidLoad()

        let button = UIButton(type: .system)
        button.setTitle("Tap", for: .normal)
        button.frame = CGRect(x: 100, y: 100, width: 100, height: 100)
        button.addTarget(self, action: #selector(iterate(_:)), for: .touchUpInside)
        view.addSubview(button)
    }

    @objc func iterate(_ sender: UIButton) {
        let tuple = generator.next()
        print(String(describing: tuple))
    }

}

PlaygroundPage.current.liveView = ViewController()

/*
 Optional((offset: 0, element: "Car"))
 Optional((offset: 1, element: "Bike"))
 Optional((offset: 2, element: "Plane"))
 Optional((offset: 3, element: "Boat"))
 nil
 nil
 nil
 */

1
获取索引是否是使用的唯一好处enumerate
亲爱的2016年

52

从Swift 2开始,需要在集合上调用枚举函数,如下所示:

for (index, element) in list.enumerate() {
    print("Item \(index): \(element)")
}

47

我在寻找使用Dictionary做到这一点的方法时找到了这个答案,事实证明,调整它非常容易,只需为元素传递一个元组即可。

// Swift 2

var list = ["a": 1, "b": 2]

for (index, (letter, value)) in list.enumerate() {
    print("Item \(index): \(letter) \(value)")
}

18

您可以简单地使用枚举循环来获得所需的结果:

斯威夫特2:

for (index, element) in elements.enumerate() {
    print("\(index): \(element)")
}

Swift 3和4:

for (index, element) in elements.enumerated() {
    print("\(index): \(element)")
}

或者,您可以简单地通过for循环来获得相同的结果:

for index in 0..<elements.count {
    let element = elements[index]
    print("\(index): \(element)")
}

希望能帮助到你。


14

基本枚举

for (index, element) in arrayOfValues.enumerate() {
// do something useful
}

或使用Swift 3 ...

for (index, element) in arrayOfValues.enumerated() {
// do something useful
}

枚举,过滤和映射

但是,我最经常将枚举与地图或过滤器结合使用。例如,对几个阵列进行操作。

在此数组中,我想过滤奇数或偶数索引元素并将其从Ints转换为Doubles。因此,enumerate()获取索引和元素,然后过滤器检查索引,最后为了摆脱生成的元组,我将其映射到元素。

let evens = arrayOfValues.enumerate().filter({
                            (index: Int, element: Int) -> Bool in
                            return index % 2 == 0
                        }).map({ (_: Int, element: Int) -> Double in
                            return Double(element)
                        })
let odds = arrayOfValues.enumerate().filter({
                            (index: Int, element: Int) -> Bool in
                            return index % 2 != 0
                        }).map({ (_: Int, element: Int) -> Double in
                            return Double(element)
                        })

9

使用.enumerate()作品,但它没有提供元素的真实索引;它仅提供一个从0开始并为每个连续元素递增1的Int。这通常是无关紧要的,但是与ArraySlice类型一起使用时,可能会发生意外行为。采取以下代码:

let a = ["a", "b", "c", "d", "e"]
a.indices //=> 0..<5

let aSlice = a[1..<4] //=> ArraySlice with ["b", "c", "d"]
aSlice.indices //=> 1..<4

var test = [Int: String]()
for (index, element) in aSlice.enumerate() {
    test[index] = element
}
test //=> [0: "b", 1: "c", 2: "d"] // indices presented as 0..<3, but they are actually 1..<4
test[0] == aSlice[0] // ERROR: out of bounds

这是一个有些人为的例子,在实践中也不是普遍的问题,但是我仍然认为有必要做到这一点。


2
it does not actually provide the true index of the element; it only provides an Int beginning with 0 and incrementing by 1 for each successive element是的,这就是为什么将其称为enumerate的原因。同样,切片不是数组,因此毫不奇怪它的行为也有所不同。这里没有错误-一切都是设计使然。:)
埃里克·艾雅

5
是的,但我从未称它为错误。对于那些不知道它如何与ArraySlice类型产生负面影响的人,我认为值得一提的是潜在的意外行为。
科尔·坎贝尔

您是否知道任何获取实际元素索引的方法-例如,如果使用filterfirst?
亚历山大·卡萨涅

9

从Swift 3开始,它是

for (index, element) in list.enumerated() {
  print("Item \(index): \(element)")
}

9

为了完整起见,您可以简单地遍历数组索引并使用下标访问相应索引处的元素:

let list = [100,200,300,400,500]
for index in list.indices {
    print("Element at:", index, " Value:", list[index])
}

使用forEach

list.indices.forEach {
    print("Element at:", $0, " Value:", list[$0])
}

使用收集enumerated()方法。请注意,它会返回带有offset和的元组集合element

for item in list.enumerated() {
    print("Element at:", item.offset, " Value:", item.element)
}

使用forEach:

list.enumerated().forEach {
    print("Element at:", $0.offset, " Value:", $0.element)
}

那些会打印

元素位于:0值:100

元素位于:1值:200

元素位于:2价值:300

元素位于:3价值:400

元素位于:4价值:500

如果需要数组索引(而不是偏移量)及其元素,则可以扩展Collection并创建自己的方法来获取索引元素:

extension Collection {
    func indexedElements(body: ((index: Index, element: Element)) throws -> Void) rethrows {
        var index = startIndex
        for element in self {
            try body((index,element))
            formIndex(after: &index)
        }
    }
}

Alex建议的另一种可能的实现方式是使用其元素压缩集合索引:

extension Collection {
    func indexedElements(body: ((index: Index, element: Element)) throws -> Void) rethrows {
        for element in zip(indices, self) { try body(element) }
    }
}

测试:

let list =  ["100","200","300","400","500"]
list.dropFirst(2).indexedElements {
    print("Index:", $0.index, "Element:", $0.element)
}

这将

索引:2元素:300

索引:3元素:400

索引:4元素:500


顺便说一句,您可以enumeratedIndices通过循环实现zip(self.indices, self)
亚历山大-恢复莫妮卡

@亚历山大-ReinstateMonica for element in zip(indices, self) { try body(element) }。顺便说一句,我不喜欢我选择的命名,indexedElements可能会更好地描述它的作用
Leo Dabus

噢,我认为这是一个更好的名字。是的,可以for循环播放,但也可以zip(self.indices, self) .forEach(body)
亚历山大-恢复莫妮卡

@ Alexander-ReinstateMonica forEach在后台进行for循环。我更愿意把它朴素简单github.com/apple/swift/blob/master/stdlib/public/core/... @inlinable public func forEach( _ body: (Element) throws -> Void ) rethrows { for element in self { try body(element) } } }
狮子座Dabus


5

我个人更喜欢使用该forEach方法:

list.enumerated().forEach { (index, element) in
    ...
}

您也可以使用简短版本:

list.enumerated().forEach { print("index: \($0.0), value: \($0.1)") }

4

Xcode 8和Swift 3:数组可以使用 tempArray.enumerated()

例:

var someStrs = [String]()

someStrs.append("Apple")  
someStrs.append("Amazon")  
someStrs += ["Google"]    


for (index, item) in someStrs.enumerated()  
{  
        print("Value at index = \(index) is \(item)").  
}

安慰:

Value at index = 0 is Apple
Value at index = 1 is Amazon
Value at index = 2 is Google

3

对于那些想要使用 forEach

斯威夫特4

extension Array {
  func forEachWithIndex(_ body: (Int, Element) throws -> Void) rethrows {
    try zip((startIndex ..< endIndex), self).forEach(body)
  }
}

要么

array.enumerated().forEach { ... }


By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.