Есть ли функция, которую я могу использовать для итерации по массиву и иметь как индекс, так и элемент, например, перечисление python?
for index, element in enumerate(list):
...
Да. Начиная с 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
может быть хорошей заменой цикла 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
*/
enumerate
?
Начиная с 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
}
Однако чаще всего я использую перечисление в сочетании с картой или фильтром. Например, при работе с парой массивов.
В этом массиве я хотел отфильтровать нечетные или даже индексированные элементы и преобразовать их из 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)
})
Вы можете просто использовать цикл перечисления, чтобы получить желаемый результат:
Свифт 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)")
}
Надеюсь, поможет.
Начиная с Swift 3, это
for (index, element) in list.enumerated() {
print("Item \(index): \(element)")
}
Это Формула цикла перечисления:
for (index, value) in shoppingList.enumerate() {
print("Item \(index + 1): \(value)")
}
для более подробной информации вы можете проверить здесь.
Использование .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
Это несколько надуманный пример, и это не общая проблема на практике, но все же я думаю, что это стоит того, чтобы знать, что это может произойти.
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 . Кроме того, срез не является массивом, поэтому неудивительно, что он ведет себя по-другому. Здесь нет ошибки - все разработано. :)
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
Для тех, кто хочет использовать 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.0), value: \($0.1)") }