回答:
はい。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)")
}
enumerate
何ですか?
enumerating
Swift 4 では、Gerundに変わるでしょう。
for (index, element) in
を使用するenumerated
と誤解を招きます。する必要がありますfor (offset, element) in
Swift 5には、と呼ばれるメソッドが用意され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.enumerated() {
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 [(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
*/
enumerate
ますか?
単純に列挙のループを使用して、希望する結果を得ることができます。
スウィフト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)")
}
それが役に立てば幸い。
for (index, element) in arrayOfValues.enumerate() {
// do something useful
}
またはSwift 3で...
for (index, element) in arrayOfValues.enumerated() {
// do something useful
}
ただし、多くの場合、enumerateをマップまたはフィルターと組み合わせて使用します。たとえば、いくつかのアレイを操作する場合です。
この配列では、奇数または偶数のインデックス付き要素をフィルター処理して、それらを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)
})
使用.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..<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と呼ばれる理由です。また、sliceは配列ではないため、動作が異なるのも当然です。ここにはバグはありません-すべてが仕様によるものです。:)
filter
最初に使用する場合
完全を期すために、配列のインデックスを反復処理し、添え字を使用して、対応するインデックスの要素にアクセスできます。
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()
方法を使用します。それはとのタプルのコレクションを返すことに注意してくださいoffset
とelement
:
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 = startIndex
for 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) }
}
}
テスト:
let list = ["100","200","300","400","500"]
list.dropFirst(2).indexedElements {
print("Index:", $0.index, "Element:", $0.element)
}
これは
インデックス:2要素:300
インデックス:3要素:400
インデックス:4要素:500
enumeratedIndices
、ループして実装できるzip(self.indices, self)
for element in zip(indices, self) { try body(element) }
。ところで、私が選んだネーミングは好きではありません。indexedElements
それが何をしているのかをよく説明しているかもしれません
for
ループは動作しますが、もzip(self.indices, self) .forEach(body)
forEach
は、舞台裏でforループを実行します。私はそれを無地シンプルに保つことを好む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) } } }
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