Свифт для цикла: для индекса, элемент в массиве?

655

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

for index, element in enumerate(list):
    ...
Теги:
arrays

12 ответов

1439
Лучший ответ

Да. Начиная с 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>.
  • 14
    Это увеличивает производительность?
Показать ещё 10 комментариев
69

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 может быть хорошей заменой цикла 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

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
 */
  • 1
    Является ли получение доступа к индексу единственным преимуществом использования enumerate ?
49

Начиная с Swift 2, функцию перечисления нужно вызвать в коллекции так:

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

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

// Swift 2

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

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

Основной перечислять

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

или со Swift 3...

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

Перечислять, фильтровать и отображать

Однако чаще всего я использую перечисление в сочетании с картой или фильтром. Например, при работе с парой массивов.

В этом массиве я хотел отфильтровать нечетные или даже индексированные элементы и преобразовать их из Ints в Double. Таким образом, 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

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

Свифт 2:

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

Свифт 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)")
}

Надеюсь, поможет.

8

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

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

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

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

для более подробной информации вы можете проверить здесь.

5

Использование .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

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

  • 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 . Кроме того, срез не является массивом, поэтому неудивительно, что он ведет себя по-другому. Здесь нет ошибки - все разработано. :)
  • 4
    Правда, но я никогда не называл это ошибкой. Это просто неожиданное поведение, которое, как я думал, стоит упомянуть для тех, кто не знает, как оно может негативно взаимодействовать с типом ArraySlice.
Показать ещё 1 комментарий
4

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

Приставки:

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

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

Swift 4

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

Или же

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

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

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

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

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

Ещё вопросы

Сообщество Overcoder
Наверх
Меню