Swift函数编程之Map、Filter、Reduce
在Swift语言中使用Map、Filter、Reduce对Array、Dictionary等集合类型(collection type)进行操作可能对一部分人来说还不是那么的习惯。对于没有接触过函数式编程的开发者来说,对集合类型中的数据进行处理的时候第一反应可能就是采用for in遍历。本文将介绍一些Swift中可以采用的新方法。
Map
Map函数会遍历集合类型并对其中的每一个元素进行同一种的操作。Map的返回值是一个所得结果的数组。例如:我们要对一个数组里面的数据进行平方操作,常见的代码如下:
let values: [Double]= [2.0,4.0,6.0,8.0]
var squares: [Double] = []
for value in values {
squares.append(value * value)
}
上面的代码中,我们使用了常规的for in遍历操作对其中的元素进行了平方操作,然后将计算的结果追加到一个变量数组里面。虽然该部分的代码很好的完成了要求,但是在Swift中我们还有更简洁和安全的代码(上面的squaers是一个变量可能出现无意的数据修改)。下面来看看使用Map进行操作的代码:
let values: [Double]= [2.0,4.0,6.0,8.0]
let squares: [Double] = values.map{$0 * $0}
该段代码不仅更加简洁而且squares是一个不可变的常量。
上面代码中的map函数的闭包语法可能对于新手比较难以理解,该闭包中只有一行对集合中数据进行处理的代码并且最终返回了结果数组。为了大家更好的理解map的操作上面的代码可以改写为:
let values: [Double]= [2.0,4.0,6.0,8.0]
let squares: [Double] = values.map({
(value: Double)-> Double in
return value * value
})
上面这段改写的代码中闭包里面传入了一个Double类型的参数,并且返回了一个相同类型的处理结果。因为map只需要一个闭包最为参数,所以我们可以使用尾数闭包的特性去除(),而且闭包里面的代码也只有一行我们可以利用单表达式的隐式返回省略return:
let squares: [Double] = values.map{value in value * value}
上面value也可以直接使用闭包的参数缩写功能给替换掉:
let squares: [Double] = values.map{$0 * $0}
map操作返回的结果数组中元素的类型并不要求与原来的元素类型一致,例如我们可以将一个常见的数字数组转为对应的单词数组:
let scores = [0,28,124]
let words = scores.map { NSNumberFormatter.localizedStringFromNumber($0, numberStyle: .SpellOutStyle) }
//["zero", "twenty-eight", "one hundred twenty-four"]
当然除了上面的Array,Set和Dictionary也能应用map操作。
Filter
Filter函数操作会对集合类型进行遍历并将其中的满足条件的元素作为结果数组中的元素进行返回。该函数里面只有一个作为条件判断的语句,闭包会遍历集合里面的元素并将满足条件的结果放在一起:
let digits = [1,4,10,15] let even = digits.filter { $0 % 2 == 0 }
// [4, 10]
Reduce
Reduce函数操作会将集合类型里面的所有元素组合成一个新值并返回。reduce中的参数为两个:一个初始值、一个combine闭包。例如下面的代码将数组中的元素相加并且其中的初始值为10:
let items = [2.0,4.0,5.0,7.0]
let total = items.reduce(10.0,combine: +)
//28.0
除了上面的数字类型之外也可以对字符串进行处理:
let codes = ["Big","nerd","coding"]
let text = codes.reduce("", combine: +)
// "Bignerdcoding"
reduce中第二个参数是一个闭包,所有你可以使用尾随闭包来自我特定操作:
let codes = ["Big","nerd","coding"]
let text = codes.reduce("v2ex") {text, name in "\(text),\(name)"}
// "v2ex,Big,nerd,coding"
FlatMap
该函数会将那些多维集合类型转换为一维集合类型,实例如下:
let collections = [[5,2,7],[4,8],[9,1,3]] let flat = collections.flatMap { $0 }
// [5, 2, 7, 4, 8, 9, 1, 3]
另外对于可选类型的集合类型来说该函数还能将其中的空值移除掉:
let codes: [String?] = ["Big",nil,"nerd",nil,"coding"]
let values = codes.flatMap {$0}
// ["Big","nerd","coding"]
真正体现flatMap强大功能的地方是与上面一个函数进行组合操作:
let collections = [[5,2,7],[4,8],[9,1,3]]
let onlyEven = collections.flatMap {
intArray in intArray.filter { $0 % 2 == 0 }
}
// [2, 4, 8]
上面的代码实现了将多维整形数组里面的偶数筛选出来并且组合成了一个一位数组。flatMap操作的参数是一个以[Int]数组作为参数的闭包。当然我们也可以使用隐含参数对其进行简写:
let collections = [[5,2,7],[4,8],[9,1,3]]
let onlyEven = collections.flatMap {
$0 in $0.filter { $0 % 2 == 0 }
}
// [2, 4, 8]
注意:上面简写中第一个和第二个$0表示collections中类似[5,2,7]的字数组,而第三个则表示子数组里面的每个整数
与其它操作进行组合的实例:
//与map操作的组合以及简写
let allSquared = collections.flatMap {
intArray in intArray.map { $0 * $0 }
}
// [25, 4, 49, 16, 64, 81, 1, 9]
let allSquared = collections.flatMap {
$0.map { $0 * $0 }
}
//与reduce操作的组合以及对等的组合操作
let sums = collections.flatMap { $0.reduce(0, combine: +) }
//对应的组合操作,两者结果是一样的
let sums = collections.map { $0.reduce(0, combine: +) }
链式组合
我们在上面已经看到了flatMap的闭包里面可以与另一操作的组合。我们还可以在闭包的外面对这些操作进行合理的组合来实现我们的目标。例如将数组中大于某个数字的所有数字进行求和操作:
let marks = [4,5,8,2,9,7]
let totalPass = marks.filter{$0 >= 7}.reduce(0,combine: +)
// 24
或者对某一个数组里面的数字进行平方操作然后在进行筛选:
let numbers = [20,17,35,4,12]
let evenSquares = numbers.map{$0 * $0}.filter{$0 % 2 == 0}
// [400, 16, 144]
总结
下次你要对集合类型的元素进行遍历并对其中的每个元素进行处理的时候,可以先检查一下是否可以直接使用上面的这些操作或者组合操作。
链接:
Swift函数编程之Map、Filter、Reduce的更多相关文章
- 高阶函数概念以及map/filter/reduce
什么样的函数叫高阶函数:map(func, *iterables) --> map object 条件:1.函数接受函数作为参数 2.函数的返回值中包含函数 num_l = [1,2,3,4,5 ...
- python 内置函数 map filter reduce lambda
map(函数名,可遍历迭代的对象) # 列组元素全加 10 # map(需要做什么的函数,遍历迭代对象)函数 map()遍历序列得到一个列表,列表的序号和个数和原来一样 l = [2,3,4,5,6, ...
- Swift map filter reduce 使用指南
转载:https://useyourloaf.com/blog/swift-guide-to-map-filter-reduce/ Using map, filter or reduce to ope ...
- python常用函数进阶(2)之map,filter,reduce,zip
Basic Python : Map, Filter, Reduce, Zip 1-Map() 1.1 Syntax # fun : a function applying to the iterab ...
- Python函数式编程之map()
Python函数式编程之map() Python中map().filter().reduce()这三个都是应用于序列的内置函数. 格式: map(func, seq1[, seq2,…]) 第一个参数 ...
- 数组的高阶方法map filter reduce的使用
数组中常用的高阶方法: foreach map filter reduce some every 在这些方法中都是对数组中每一个元素进行遍历操作,只有foreach是没有 ...
- 如何在python3.3用 map filter reduce
在3.3里,如果直接使用map(), filter(), reduce(), 会出现 >>> def f(x): return x % 2 != 0 and x % 3 != 0 ...
- Python map,filter,reduce函数
# -*- coding:utf-8 -*- #定义一个自己的map函数list_list = [1,2,4,8,16] def my_map(func,iterable): my_list = [] ...
- 高阶函数map(),filter(),reduce()
接受函数作为参数,或者把函数作为结果返回的函数是高阶函数,官方叫做 Higher-order functions. map()和filter()是内置函数.在python3中,reduce()已不再是 ...
随机推荐
- Android NDK目录介绍
交叉编译 在一个平台上去编译另一个平台上可以执行的本地代码 cpu平台---arm x86 mips 操作系统平台---windows linux mac os 原理 模拟不同平台的特性去编译代码 j ...
- Android对话框
这周过的实在是艰辛,自打这周二起我的本本就开始闹"罢工",最后还是重装系统了事. . . 只是可怜了我的那些被格了的软件(悲伤辣么大)! 往事不要再提,人生几度风雨... 简 ...
- UIscrollView和UIPageControl的循环滚动
因为昨天在网上找了很久,很多只能实现向右滚动,而且一张图一个imageview ,感觉工作量很可怕啊 , 下面的例子就是不论你多少图 , 只和我代码里面的几个数值有关, 只需要修改分页和循环i的最 ...
- php文件下载
public function down() { $lang = strtolower(cookie('think_language')); if ($lang == 'en-us') { $file ...
- Java基础知识学习(八)
IO操作 5个重要的类分别是:InputStream.OutStream.Reader.Writer和File类 面向字符的输入输出流 输入流都是Reader的子类, CharArrayReader ...
- nodejs学习笔记(1)--express安装问题:express不是内部也或者外部的命令解决方案
"Express 是一个简洁而灵活的 node.js Web应用框架, 提供了一系列强大特性帮助你创建各种 Web 应用,和丰富的 HTTP 工具.使用 Express 可以快速地搭建一个完 ...
- EnumMap
以下内容基于jdk1.7.0_79源码: 什么是EnumMap Map接口的实现,其key-value映射中的key是Enum类型: 补充说明 其原理就是一个对象数组,数组的下标索引就是根据Map中的 ...
- Java注解一谈
阅读目录 1.元注解 2.自定义注解 3.注解处理器 android注解框架解析 我们经常会在java代码里面看到:“@Override”,“@Target”等等样子的东西,这些是什么? 在java里 ...
- SQL Server 连接超时案例一则
上周六,一工厂系统管理员反馈一数据库连接不上,SSMS连接数据库报"连接超时时间已到.在尝试使用预登录握手确认时超过了此超时时间.......", 如下截图所示: 另外远程连接也连 ...
- [20130704] Intra-Query Parallel Thread Deadlocks
今天碰到了 Intra-Query Parallel Thread Deadlocks 简单的说就是并发查询把自己给锁住了. 原理: 在并发查询运行是,有一个生产者和一个消费者,生产者等待消费者产生 ...