Arrays 如何在Swift中使用索引和元素迭代循环
是否有一个函数可以用来迭代数组并同时具有索引和元素,比如Python的Arrays 如何在Swift中使用索引和元素迭代循环,arrays,swift,enumeration,Arrays,Swift,Enumeration,是否有一个函数可以用来迭代数组并同时具有索引和元素,比如Python的枚举 for index, element in enumerate(list): ... 对。从Swift 3.0开始,如果需要每个元素的索引及其值,可以使用来迭代数组。它返回由索引和数组中每个项的值组成的成对序列。例如: for (index, element) in list.enumerated() { print("Item \(index): \(element)") } let list = ["
枚举
for index, element in enumerate(list):
...
对。从Swift 3.0开始,如果需要每个元素的索引及其值,可以使用来迭代数组。它返回由索引和数组中每个项的值组成的成对序列。例如:
for (index, element) in list.enumerated() {
print("Item \(index): \(element)")
}
let list = ["Car", "Bike", "Plane", "Boat"]
for (index, element) in list.enumerated() {
print(index, ":", element)
}
/*
prints:
0 : Car
1 : Bike
2 : Plane
3 : Boat
*/
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
*/
在Swift 3.0之前和Swift 2.0之后,该函数被调用为enumerate()
:
在Swift 2.0之前,enumerate
是一个全局函数
for (index, element) in enumerate(list) {
println("Item \(index): \(element)")
}
我在寻找一种使用字典实现这一点的方法时找到了这个答案,结果证明它很容易适应,只需为元素传递一个元组
// Swift 2
var list = ["a": 1, "b": 2]
for (index, (letter, value)) in list.enumerate() {
print("Item \(index): \(letter) \(value)")
}
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)
})
从Swift 2开始,需要对集合调用枚举函数,如下所示:
for (index, element) in list.enumerate() {
print("Item \(index): \(element)")
}
Swift 5提供了一种称为
数组
的方法<代码>枚举()具有以下声明:
func enumerated() -> EnumeratedSequence<Array<Element>>
但是请注意,您不限于将
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)]
或者,您也可以将
枚举()
与映射一起使用:
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()
,您甚至可以在数组上手动迭代。例如:
for (index, element) in list.enumerated() {
print("Item \(index): \(element)")
}
let list = ["Car", "Bike", "Plane", "Boat"]
for (index, element) in list.enumerated() {
print(index, ":", element)
}
/*
prints:
0 : Car
1 : Bike
2 : Plane
3 : Boat
*/
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
*/
这是枚举循环的公式:
for (index, value) in shoppingList.enumerate() {
print("Item \(index + 1): \(value)")
}
有关更多详细信息,请查看。Basic enumerate
或与斯威夫特3
for (index, element) in arrayOfValues.enumerated() {
// do something useful
}
枚举、筛选和映射
然而,我最常将枚举与映射或过滤器结合使用。例如,对几个阵列进行操作
在这个数组中,我想过滤奇数或偶数索引元素,并将它们从int转换为double。因此,enumerate()
获取索引和元素,然后过滤器检查索引,最后为了去掉生成的元组,我将其映射到元素
// Swift 2
var list = ["a": 1, "b": 2]
for (index, (letter, value)) in list.enumerate() {
print("Item \(index): \(letter) \(value)")
}
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的整数。这通常无关紧要,但与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
设a=[“a”、“b”、“c”、“d”、“e”]
a、 指数/=>0。。1.[0:“b”,1:“c”,2:“d”]//指数以0表示。从Swift 3开始,它是
for (index, element) in list.enumerated() {
print("Item \(index): \(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
Swift 5.x:
我个人更喜欢使用forEach
方法:
list.enumerated().forEach { (index, element) in
...
}
您也可以使用短版本:
list.enumerated().forEach { print("index: \($0.0), value: \($0.1)") }
您只需使用枚举循环即可获得所需的结果:
Swift 2:
for (index, element) in elements.enumerate() {
print("\(index): \(element)")
}
for (index, element) in elements.enumerated() {
print("\(index): \(element)")
}
Swift 3&4:
for (index, element) in elements.enumerate() {
print("\(index): \(element)")
}
for (index, element) in elements.enumerated() {
print("\(index): \(element)")
}
或者,您可以简单地通过for循环获得相同的结果:
for index in 0..<elements.count {
let element = elements[index]
print("\(index): \(element)")
}
用于0中的索引。用于那些想要使用forEach
的人
斯威夫特4
extension Array {
func forEachWithIndex(_ body: (Int, Element) throws -> Void) rethrows {
try zip((startIndex ..< endIndex), self).forEach(body)
}
}
为了完整性,您只需迭代数组索引并使用下标访问相应索引处的元素:
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])
}
使用集合枚举()
方法。请注意,它返回一个元组集合,其中包含偏移量
和元素
:
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)
}
}
}
extension Collection {
func indexedElements(body: ((index: Index, element: Element)) throws -> Void) rethrows {
for element in zip(indices, self) { try body(element) }
}
var indexedElements: Zip2Sequence<Indices, Self> { zip(indices, self) }
}
Alex建议的另一种可能的实现是压缩集合索引及其元素:
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) }
}
var indexedElements: Zip2Sequence<Indices, Self> { zip(indices, self) }
}
这将是p[rint
索引:2元素:300
索引:3元素:400
索引:4元素:500
索引:0元素:100
索引:1元素:200
索引:2元素:300
索引:3元素:400
索引:4元素:500
我们调用枚举函数来实现这个
for (index, element) in array.enumerate() {
index is indexposition of array
element is element of array
}
对于要执行的操作,应在数组上使用enumerated()
方法:
for (index, element) in list.enumerated() {
print("\(index) - \(element)")
}
在函数式编程中像这样使用.enumerated():
list.enumerated().forEach { print($0.offset, $0.element) }
虽然它看起来像一个元组,但在Swift 1.2中-不确定是否为2.0-enumerate返回EnumerateSequence结构。@Leviathlon显著或可测量的性能开销是否重要?否。可能它们会更改为动名词,枚举
用于Swift 4。激动人心!@Honeyfor(索引,元素)在
中使用
枚举时
具有误导性。应为(偏移量,元素)在
中,我们是否会举行抽奖,看谁能猜出它下一步会变成什么。这种该死的语言就是坐不住。lol它实际上并没有提供元素的真正索引;它只提供一个以0开头的整数,每个后续元素递增1
是的,这就是为什么它被称为enumerate。而且,slice不是因此,毫不奇怪它会有不同的行为。这里没有bug-一切都是设计的。:)没错,但我从来没有把它称为bug。对于那些不知道它如何与ArraySlice类型产生负面影响的人来说,这只是一种潜在的意外行为,我认为值得一提。你知道有什么方法可以获得实际e的索引吗元素-例如,如果先使用过滤器
?访问索引是使用枚举
的唯一好处吗?顺便说一句,您可以通过使用zip(self.indexes,self)
对zip(index,self){try body中的元素进行循环来实现EnumeratedDices
(