swift for loop: для индекса, элемента в массиве?

есть ли функция, которую я могу использовать для итерации по массиву и иметь как индекс, так и элемент, например, перечисление python?

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

12 ответов


да. Начиная с 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)")
}

Swift 3 предоставляет метод под названием enumerated() на Array. enumerated() есть следующее объявление:

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

возвращает последовательность пар (n, x), где n представляет последовательное целое число, начинающееся с нуля, а x представляет элемент последовательности.


в самых простых случаях вы можете использовать enumerated() С помощью цикла for. Например:

let list = ["Car", "Bike", "Plane", "Boat"]
for (index, element) in list.enumerate() {
    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 [(0, 1), (1, 2), (2, 3), (3, 4), (4, 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 может быть хорошей заменой для петля:

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

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

С помощью enumerated() и makeIterator(), вы даже можете повторить вручную на вашем Array. Например:

import UIKit

class ViewController: UIViewController {

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

    // Link this IBAction to a UIButton in your storyboard
    @IBAction func iterate(_ sender: UIButton) {
        let tuple: (offset: Int, element: String)? = generator.next()
        print(String(describing: tuple))
    }

}

/*
 Will print the following lines for 6 `touch up inside`:
 Optional((0, "Car"))
 Optional((1, "Bike"))
 Optional((2, "Plane"))
 Optional((3, "Boat"))
 nil
 nil
 */

начиная с Swift 2, Функция перечисления должна вызываться в коллекции следующим образом:

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

Я нашел этот ответ, ища способ сделать это с помощью словарь, и оказывается, его довольно легко адаптировать, просто передайте Кортеж для элемента.

// Swift 2

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

for (index, (letter, value)) in list.enumerate() {
    print("Item \(index): \(letter) \(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() получает индекс и элемент, затем фильтр проверяет индекс и, наконец, чтобы избавиться от полученного кортеж I сопоставляет его только с элементом.

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

начиная с Swift 3, это

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

Это формула цикла перечисления:

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

для Больше детали вы можете проверить здесь.


вы можете просто использовать цикл перебора, чтобы получить желаемый результат:

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

надеюсь, что это помогает.


используя .enumerate() работает, но он не предоставляет истинный индекс элемента; он предоставляет только Int, начинающийся с 0 и увеличивающийся на 1 для каждого последующего элемента. Обычно это не имеет значения, но при использовании 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

это несколько надуманный пример, и это не обычная проблема на практике, но все же я думаю, что стоит знать, что это может произойти.


Xcode 8 и Swift 3: Массив можно перечислить с помощью tempArray.enumerate ()

пример:

var someStrs = [String]()

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


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

для тех, кто хочет использовать forEach.

Swift 4

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

или

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

Я лично предпочитаю использовать forEach способ:

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

вы также можете использовать короткую версию:

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