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

是否有一个函数可以用来迭代数组并同时具有索引和元素,就像Python的enumerate一样?

for index, element in enumerate(list):...
617494 次浏览

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

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)")}

我在寻找使用字典的方法时找到了这个答案,事实证明,调整它很容易,只需为元素传递一个元组。

// Swift 2
var list = ["a": 1, "b": 2]
for (index, (letter, value)) in list.enumerate() {print("Item \(index): \(letter) \(value)")}

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

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

Swift 5为Array提供了一个名为enumerated()的方法。enumerated()有以下声明:

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 : Car1 : Bike2 : Plane3 : 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 : 51 : 42 : 33 : 24 : 1*/

通过使用enumerated()makeIterator(),您甚至可以手动迭代Array。例如:

import UIKitimport 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"))nilnilnil*/

这是枚举循环的公式:

for (index, value) in shoppingList.enumerate() {print("Item \(index + 1): \(value)")}

有关详细信息,您可以查看这里

基本枚举

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

或者使用Swift 3…

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

枚举、过滤和映射

但是,我经常将enumerate与map或filter结合使用。例如,对几个数组进行操作。

在这个数组中,我想过滤奇数或偶数索引元素并将它们从Ints转换为Double。所以enumerate()获取索引和元素,然后过滤器检查索引,最后为了摆脱结果元组,我将其映射到元素。

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

使用.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..<4test[0] == aSlice[0] // ERROR: out of bounds

这是一个有点做作的例子,在实践中这不是一个常见的问题,但我仍然认为值得知道这可能发生。

从Swift 3开始,它是

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

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 AppleValue at index = 1 is AmazonValue at index = 2 is Google

Swift 5. x:

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

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

您也可以使用简短版本:

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

你可以简单地使用枚举循环来获得你想要的结果:

Swift 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)")}

希望有帮助。

对于那些想要使用forEach的人。

Swift4

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

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

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

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()方法。请注意,它返回带有offsetelement的元组集合:

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 = startIndexfor 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) }}var indexedElements: Zip2Sequence<Indices, Self> { zip(indices, self) }}

测试:

let list =  ["100","200","300","400","500"]
// You can iterate the index and its elements using a closurelist.dropFirst(2).indexedElements {print("Index:", $0.index, "Element:", $0.element)}
// or using a for loopfor (index, element) in list.indexedElements  {print("Index:", index, "Element:", element)}

这将p[rint]

索引:2元素:300

索引:3元素:400

指数:4元素:500

索引:0元素:100

索引:1元素:200

索引:2元素:300

索引:3元素:400

指数:4元素:500

我们调用enumerate函数来实现这个。喜欢

    for (index, element) in array.enumerate() {index is indexposition of arrayelement is element of array}

对于你想要做的事情,你应该在你的数组上使用enumerated()方法:

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

在函数式编程中像这样使用. enumerated():

list.enumerated().forEach { print($0.offset, $0.element) }

Swift 5. x:

让列表=[0,1,2,3,4,5]

list.enumerated().forEach { (index, value) inprint("index: \(index), value: \(value)")}

或者,

list.enumerated().forEach {print("index: \($0.offset), value: \($0.element)")}

或者,

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

iOS8.0/Swift4.0+

您可以使用forEach苹果文档

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

let numberWords = ["one", "two", "three"]
numberWords.enumerated().forEach { (key, value) inprint("Key: \(key) - Value: \(value)")}