Swiftでインデックスと要素を使ってループを繰り返す方法


784

Pythonの列挙のように、配列を反復処理してインデックスと要素の両方を持つことができる関数はありますか?

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

回答:


1661

はい。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>構造体を返します。
nstein 2015

2
@Leviathlon気になる、または測定可能なパフォーマンスのオーバーヘッドは重要ですか?No.
Dan Rosenstark、2016年

インデックスへのアクセスを使用することの唯一の利点はenumerate何ですか?
Honey

29
たぶん、enumeratingSwift 4 では、Gerundに変わるでしょう。
Dan Rosenstark 2017年

3
@Honey for (index, element) inを使用するenumeratedと誤解を招きます。する必要がありますfor (offset, element) in
レオダバス

116

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
 */

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
}

列挙、フィルター、マップ

ただし、多くの場合、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)
                        })

9

使用.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

これはやや不自然な例であり、実際には一般的な問題ではありませんが、これが発生する可能性があることを知っておく価値はあります。


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と呼ばれる理由です。また、sliceは配列ではないため、動作が異なるのも当然です。ここにはバグはありません-すべてが仕様によるものです。:)
Eric Aya

5
本当ですが、私はそれをバグとは呼びませんでした。ArraySlice型とどのように悪影響を及ぼし合うのかを知らない人のために言及する価値があると思ったのは、潜在的に予期しない動作です。
コールキャンベル

実際の要素のインデックスを取得する方法を知っていますか-たとえばfilter最初に使用する場合
Alexandre Cassagne


9

完全を期すために、配列のインデックスを反復処理し、添え字を使用して、対応するインデックスの要素にアクセスできます。

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()方法を使用します。それはとのタプルのコレクションを返すことに注意してくださいoffsetelement

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)
アレクサンダー-モニカを復活させる

@アレクサンダー・ReinstateMonica for element in zip(indices, self) { try body(element) }。ところで、私が選んだネーミングは好きではありません。indexedElementsそれが何をしているのかをよく説明しているかもしれません
レオダバス

うーん、いい名前だと思います。うん、forループは動作しますが、もzip(self.indices, self) .forEach(body)
復活モニカ-アレクサンダー

@ Alexander-ReinstateMonica 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) } } }
レオDabus

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

これを実装するために列挙関数を呼び出しました。お気に入り

(index、element)for array.enumerate(){

//インデックスは配列のインデックス位置です

//要素は配列の要素です

}

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.