Swift 做为一门新兴的语言,它吸取了众多语言的优势,函数式编程就是其中之一。在 Swift 中,函数是做为一等公民
的存在,因此学习高阶函数仍是很是有必要的,它可使你的代码扩展性更高,代码更 Swift 范。废话很少说,下面就让咱们开始吧!编程
在 Swift 中,高阶函数一共有下面几个:swift
合并操做,
而后将合并操做后的结果放置在数组中返回。非空的
映射结果放置在数组中返回。非空的
映射结果-键值对放置在字典中返回。符合条件的元素
放在数组中返回。合并
,并将合并结果返回。经过上面的阐述,咱们大概了解到了这几个函数是作什么用的,下面咱们在经过几个例子来具体看一下代码上如何使用。数组
对于 map 函数,使用场景就是将数组的类型映射为别的类型。好比,咱们有一个模型数组,模型的 id 字段咱们从服务器拿的是 String 类型,在某种场景下咱们须要转为 Int 类型,这时候咱们就能够经过 map 函数来实现该需求。bash
struct Student {
let id: String
let name: String
let age: Int
}
let stu1 = Student(id: "1001", name: "stu1", age: 12)
let stu2 = Student(id: "1002", name: "stu2", age: 14)
let stu3 = Student(id: "1003", name: "stu3", age: 16)
let stu4 = Student(id: "1004", name: "stu4", age: 20)
let stus = [stu1, stu2, stu3, stu4]
let intIds = stus.map { (stu) in
Int(stu.id)
}
print(intIds) //[Optional(1001), Optional(1002), Optional(1003), Optional(1004)]
复制代码
经过上述代码,咱们将 id 字段从 String 映射为了 Int? 类型,这并非咱们想要的 Int 类型。若是咱们须要访问元素的话还得须要解包,那么咱们如何既能将元素映射又能自动筛选 nil 的值呢?这时,就轮到 compactmap 出马了。服务器
optional 也可使用map函数
闭包
var num: Int? = 2
let result = num.map {
$0 * 2
}
print(result) // Optional(4)
// ----而不是使用下面的代码
var result: Int?
if let n = num {
result = n * 2
} else {
result = nil
}
复制代码
咱们将上面的代码替换为:app
let intIds = stus.compactMap { (stu) in
Int(stu.id)
}
复制代码
这时,咱们再打印 intIds 就会发现它已经为 Int 类型了。函数式编程
对于 Set 和 Array ,你可使用 compactMap 来得到非空的集合,可是对于 Dictionary 来讲,这个函数是不起做用的。函数
let dict = ["key1": 10, "key2": nil]
let result = dict.compactMap { $0 }
print(result) //[(key: "key1", value: Optional(10)), (key: "key2", value: nil)]
复制代码
这时候,咱们须要使用 compactMapValues 函数来得到非空的字典。学习
let dict = ["key1": 10, "key2": nil]
let result = dict.compactMapValues { $0 }
print(result) //["key1": 10]
复制代码
对于 flatMap,主要的应用场景就是你想得到一个单层集合的数组。经过下面的代码来看一下 map 和 flapMap 的区别。
let scoresByName = ["Henk": [0, 5, 8], "John": [2, 5, 8]]
let mapped = scoresByName.map { $0.value }
// [[2, 5, 8], [0, 5, 8]]
print(mapped)
let flatMapped = scoresByName.flatMap { $0.value }
// [2, 5, 8, 0, 5, 8]
复制代码
map 会直接将元素放在数组中,而 flatMap 会将元素平铺在一个数组中。实际上,s.flatMap(transform)
等同于s.map(transform).joined()
。
这个函数就如同单词的意思:查找。将符合条件的元素查找出来放置在数组中返回。好比咱们想查找年龄大于18岁的全部学生。
let adults = stus.filter { (stu) -> Bool in
stu.age >= 18
}
print(adults) // 数组中只包含stu4 学生
复制代码
对于 reduce,咱们的使用场景就是对数组中的元素进行组合运算,好比咱们想计算全部学生的年龄加载一块儿是多少。
let totalAges = stus.reduce(0) { (result, stu) in
return result + stu.age
}
print(totalAges) // 62
复制代码
该函数的第一个参数为初始值,后面元组中的第一个参数为每次计算的结果,第二个参数为每次遍历的元素。最后将计算的结果返回。
对于使用高阶函数最大的好处就是能够进行函数式编程了。下面咱们经过几个小栗子来对这几个高阶函数进行组合使用。
let adults = stus.compactMap { (stu) in
Int(stu.id)
}.filter { (id) -> Bool in
id > 1002
}
print(adults) //[1003, 1004]
复制代码
let totalAge = stus.filter { (stu) -> Bool in
stu.age > 12
}.reduce(0) { (result, stu) in
return result + stu.age
}
print(totalAge) // 50
复制代码
经过上面的讲述咱们知道了这几个函数的工做原理,下面咱们来动手本身实现如下这几个函数,加深一下对函数的理解。
下面的几个函数只是实现一下大致思路,并无很细致的实现官方函数的功能。
func customMap<T>(translate: (Element) -> T) -> [T] {
var results = [T]()
self.forEach { (ele) in
results.append(translate(ele))
}
return results
}
复制代码
func customCompactMap<T>(translate: (Element) -> T?) -> [T] {
var results = [T]()
for ele in self {
if let value = translate(ele) {
results.append(value)
}
}
return results
}
复制代码
func customFilter(condition: (Element) -> Bool) -> [Element] {
var results = [Element]()
self.forEach { (ele) in
if condition(ele) {
results.append(ele)
}
}
return results
}
复制代码
func customReduce<T>(initialvalue: T, produce:(T, Element) -> T) -> T {
var total = initialvalue
self.forEach { (ele) in
total = produce(total, ele)
}
return total
}
复制代码
思惟导图: