语言指南没有显示列表理解的痕迹。在Swift中完成这个的最简单的方法是什么?我在寻找类似的东西:
evens = [ x for x in range(10) if x % 2 == 0]
从Swift 2.x开始,有一些与你的Python样式列表解析相当的东西。
Python的公式(它读取类似“将变换应用到受过滤器处理的序列”)的最直接的修改涉及链接所有SequenceTypes可用的映射和过滤器方法,并从范围开始:
// Python: [ x for x in range(10) if x % 2 == 0 ] let evens = (0..<10).filter { $0 % 2 == 0 } // Another example,since the first with 'x for x' doesn't // use the full ability of a list comprehension: // Python: [ x*x for x in range(10) if x % 2 == 0 ] let evenSquared = (0..<10).filter({ $0 % 2 == 0 }).map({ $0 * $0 })
注意,Range是抽象的 – 它实际上并不创建你所要求的值的整个列表,只是一个按需提供它们的结构。 (在这个意义上,它更像是Python的xrange。)然而,过滤器调用返回一个Array,所以你失去了“lazy”方面。如果你想保持集合懒惰一路通过,只是这样说:
// Python: [ x for x in range(10) if x % 2 == 0 ] let evens = (0..<10).lazy.filter { $0 % 2 == 0 } // Python: [ x*x for x in range(10) if x % 2 == 0 ] let evenSquared = (0..<10).lazy.filter({ $0 % 2 == 0 }).map({ $0 * $0 })
与Python中的列表解析语法(以及一些其他语言中的类似构造)不同,Swift中的这些操作遵循与其他操作相同的语法。也就是说,对一系列数字进行构造,过滤和操作的语法风格是相同的,因为它是对对象数组进行过滤和操作 – 您不必对某种工作使用函数/方法语法和列表推导语法。
您可以将其他函数传递给过滤器和地图调用,并在其他方便的转换(如sort和reduce)中传递其他函数:
// func isAwesome(person: Person) -> Bool // let people: [Person] let names = people.filter(isAwesome).sort(<).map({ $0.name }) let sum = (0..<10).reduce(0,combine: +)
然而,根据你的目的,可能会有更简洁的方法来表达你的意思。例如,如果你特别想要一个偶数的列表,你可以使用stride:
let evenStride = 0.stride(to: 10,by: 2) // or stride(through:by:),to include 10
与范围一样,这会得到一个生成器,所以你需要从它创建一个数组或迭代它来查看所有的值:
let evensArray = Array(evenStride) // [0,2,4,6,8]
编辑:重新修改Swift 2.x.如果你想要Swift 1.x,请参阅编辑历史。