Як повторити цикл з індексом та елементом у Swift


784

Чи є функція, яку я можу використовувати, щоб перебирати масив і мати як індекс, так і елемент, як перерахування Python?

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

Відповіді:


1660

Так. Що стосується 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 - перераховує повертає EnumerateSequence <base: SequenceType> struct.
nstein

2
@ Leviathlon помітні або вимірювані ефективні накладні витрати, що буде мати значення? №
Дан Розенстарк

чи є доступ до індексу єдиною перевагою використання enumerate?
Мед

29
Можливо, вони зміняться на герунду, enumeratingдля Swift 4. Захоплююче!
Дан Розенстарк

3
@Honey for (index, element) inпри використанні enumeratedвводить в оману. має бутиfor (offset, element) in
Лев Дабус

115

Swift 5 пропонує метод, enumerated()для якого вимагається Array. enumerated()має таку заяву:

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

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


У найпростіших випадках ви можете використовувати enumerated()для циклу. Наприклад:

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()для циклу. Насправді, якщо ви плануєте використовувати 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?
Мед

52

Починаючи з Swift 2, функцію перерахування потрібно викликати у колекції так:

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

47

Я знайшов цю відповідь, шукаючи спосіб зробити це зі словником , і виявляється, адаптувати його досить просто, просто передати кортеж для елемента.

// 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 в парні. Так 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()робіт, але це не дає справжнього індексу елемента; він забезпечує лише 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Так, саме тому його називають перелічуванням . Також фрагмент не є масивом, тому не дивно, він поводиться інакше. Тут немає ніяких помилок - все за задумом. :)
Ерік Айя

5
Щоправда, але я ніколи не називав це помилкою. Це якраз потенційно несподівана поведінка, яку я вважав, варто згадати тим, хто не знав, як це може негативно взаємодіяти з типом ArraySlice.
Коул Кемпбелл

Чи знаєте ви будь-яким способом отримати індекс фактичного елемента - наприклад, якщо використовується filterперший?
Олександр Кассань


9

Для повноти ви можете просто повторити індекси масиву та використовувати підписник для доступу до елемента у відповідному індексі:

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

Використання дляEach

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

за допомогоюEach:

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

Ці надрукують

Елемент у: 0 Значення: 100

Елемент у: 1 Значення: 200

Елемент у: 2 Значення: 300

Елемент у: 3 Значення: 400

Елемент у: 4 Значення: 500

Якщо вам потрібен індекс масиву (а не зсув) та його елемент, ви можете розширити колекцію та створити власний метод для отримання індексованих елементів:

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

Ще одна можлива реалізація, як запропонував Алекс, - це зібрати індекси колекції з її елементами:

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

Це буде p [rint

Покажчик: 2 елемент: 300

Покажчик: 3 елемент: 400

Індекс: 4 елемент: 500


BTW ви можете реалізувати enumeratedIndices, переглянувшиzip(self.indices, self)
Олександр - Відновіть Моніку

@ Alexander-ReinstateMonica for element in zip(indices, self) { try body(element) }. До речі, мені не подобається indexedElements
вибране нами

Оо, я думаю, це краще ім’я. Так, forпетля працює, але такожzip(self.indices, self) .forEach(body)
Олександр - Відновіть Моніку

@ Alexander-ReinstateMonica forEachробить цикл за кадром. Я вважаю за краще просто використовувати 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) } } }
Лев Дабус

7

Це формула циклу перерахунку:

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

для більш детальної інформації ви можете перевірити тут .


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 { ... }

2

Для того, що ви хочете зробити, ви повинні використовувати enumerated()метод у своєму масиві :

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

-1

Ми закликали функцію перерахування, щоб реалізувати це. подібно до

для (індекс, елемент) в array.enumerate () {

// index - це індексація масиву

// елемент - це елемент масиву

}

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.