วิธีการวนซ้ำกับดัชนีและองค์ประกอบใน Swift


784

มีฟังก์ชั่นที่ฉันสามารถใช้วนซ้ำอาร์เรย์และมีทั้งดัชนีและองค์ประกอบเช่น Python enumerate หรือไม่?

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
แม้ว่ามันจะดูเหมือนเป็น tuple แต่ใน Swift 1.2 - ไม่แน่ใจเกี่ยวกับ 2.0 - enumerate ส่งคืน EnumerateSequence <base: SequenceType> struct
nstein

2
@Leviathlon ค่าใช้จ่ายที่สังเกตเห็นได้ชัดเจนหรือวัดได้ซึ่งจะเป็นเรื่องสำคัญ ไม่
Dan Rosenstark

การเข้าถึงดัชนีมีประโยชน์เพียงอย่างเดียวของการใช้งานenumerateหรือไม่
ฮันนี่

29
บางทีพวกเขาอาจจะเปลี่ยนเป็นคำนามenumeratingสำหรับ Swift 4 ที่น่าตื่นเต้น!
Dan Rosenstark

3
@ เงินfor (index, element) inเมื่อใช้enumeratedถูกทำให้เข้าใจผิด ควรเป็นfor (offset, element) in
Leo Dabus

115

สวิฟท์ 5 มีวิธีที่เรียกว่าสำหรับenumerated() มีการประกาศดังต่อไปนี้:Arrayenumerated()

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()กับลูป 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สามารถทดแทนที่ดีที่จะสำหรับห่วง:

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 ฟังก์ชัน enumerate ต้องถูกเรียกในคอลเล็กชันดังนี้:

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

47

ฉันพบคำตอบนี้ในขณะที่กำลังมองหาวิธีที่จะทำเช่นนั้นกับพจนานุกรมและปรากฎว่ามันค่อนข้างง่ายที่จะปรับมันเพียงแค่ส่ง tuple สำหรับองค์ประกอบ

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

สวิฟท์ 3 และ 4:

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

หรือคุณเพียงแค่ผ่าน for for loop เพื่อให้ได้ผลลัพธ์เดียวกัน:

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 เป็น Doubles เพื่อให้enumerate()คุณได้รับดัชนีและองค์ประกอบจากนั้นตัวกรองจะตรวจสอบดัชนีและในที่สุดก็กำจัด tuple ที่เกิดขึ้นที่ฉันจับคู่กับองค์ประกอบ

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ใช่ว่าทำไมมันเรียกว่าการระบุ อีกทั้งชิ้นส่วนยังไม่ได้เรียงกันดังนั้นจึงไม่แปลกที่มันจะทำงานแตกต่างกัน ไม่มีข้อบกพร่องที่นี่ - ทุกอย่างเกิดจากการออกแบบ :)
Eric Aya

5
จริง แต่ฉันไม่เคยเรียกมันว่าบั๊ก มันเป็นพฤติกรรมที่คาดไม่ถึงที่ฉันคิดว่ามันคุ้มค่าที่จะพูดถึงสำหรับคนที่ไม่รู้ว่ามันจะมีปฏิสัมพันธ์ในเชิงลบกับประเภท ArraySlice ได้อย่างไร
Cole Campbell

คุณตระหนักถึงวิธีการรับดัชนีขององค์ประกอบจริง - เช่นถ้าใช้filterครั้งแรก?
Alexandre Cassagne


9

เพื่อความสมบูรณ์คุณสามารถทำซ้ำดัชนีของอาร์เรย์และใช้ตัวห้อยเพื่อเข้าถึงองค์ประกอบที่ดัชนีที่เกี่ยวข้อง:

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

ใช้สำหรับแต่ละ

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

หากคุณต้องการดัชนีอาร์เรย์ (ไม่ใช่ออฟเซ็ต) และองค์ประกอบคุณสามารถขยายการรวบรวมและสร้างวิธีการของคุณเองเพื่อรับองค์ประกอบดัชนี:

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

นี่จะ p [rint

ดัชนี: 2 องค์ประกอบ: 300

ดัชนี: 3 องค์ประกอบ: 400

ดัชนี: 4 องค์ประกอบ: 500


BTW คุณสามารถใช้งานได้enumeratedIndicesโดยวนลูปด้วยzip(self.indices, self)
Alexander - Reinstate Monica

for element in zip(indices, self) { try body(element) }@ อเล็กซานเด-ReinstateMonica Btw ฉันไม่ชอบการตั้งชื่อที่ฉันเลือกindexedElementsอาจอธิบายได้ดีขึ้นว่ามันทำอะไร
Leo Dabus

Ooo ฉันคิดว่านั่นเป็นชื่อที่ดีกว่า ใช่forวงทำงานได้ แต่zip(self.indices, self) .forEach(body)
Alexander - Reinstate Monica

@ 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) } } }
Leo 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: Array สามารถระบุได้โดยใช้ 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()วิธีนี้กับArray :

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

-1

เราเรียกใช้ฟังก์ชันแจกแจงเพื่อดำเนินการนี้ ชอบ

สำหรับ (ดัชนีองค์ประกอบ) ใน array.enumerate () {

// index คือการจัดทำดัชนีของอาร์เรย์

// องค์ประกอบเป็นองค์ประกอบของอาร์เรย์

}

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.