一、关于闭包
Swift函数是一等公民,Swift一大特性就是使用简洁的语言去替代复杂的函数操作。
所谓闭包就是自包含的函数代码块,可以在代码中被传递和使用。
Swift中的闭包与C、OC中的blcok和其他语言中的匿名函数相似。
"闭包可以捕获和存储其所在上下文中任意常量和变量的引用。被称为包裹常量和变量。 Swift 会为你管理在捕获过程中涉及到的所有内存操作。”
--------------------------------------摘录来自: “The Swift Programming Language 中文版”。
闭包采取如下三种形式之一:
- 全局函数是一个有名字但不会捕获任何值的闭包
- 嵌套函数是一个有名字并可以捕获其封闭函数域内值的闭包
- 闭包表达式是一个利用轻量级语法所写的可以捕获其上下文中变量或常量值的匿名闭包
Swift 的闭包表达式拥有简洁的风格,并鼓励在常见场景中进行语法优化,主要优化如下:
- 利用上下文推断参数和返回值类型
- 隐式返回单表达式闭包,即单表达式闭包可以省略 return 关键字
- 参数名称缩写
- 尾随闭包语法”
从繁到简看Swif中的闭包形式
//求两个整数之和
func addFun(a:Int , b:Int) -> Int{
return a + b
}
//求两个整数之积
func mulFun(a:Int , b:Int) -> Int{
return a * b
}
//对两个数进行运算 为了动态计算我们传入规则
func caculateFun(a:Int , b:Int , someFun:(Int,Int) ->Int) -> Int{
return someFun(a,b)
}
//之前我们通常会这样做:
caculateFun(a: 10, b: 2, someFun: addFun) // 12
caculateFun(a: 10, b: 2, someFun: mulFun) // 20
//然后我们可以优化成这样:
caculateFun(a: 20, b: 2, someFun: { (x, y) -> Int in
return x * y
})
//在Swift中我们可以继续我们成更改形式
//我们称之为 尾随闭包
caculateFun(a: 20, b: 2) { (x, y) -> Int in
return x * y
}//40
//又因为Swift中存在类型推断 能从caculateFun的声明中推断出传入作为参数的函数期望接收两个Int并返回一个Int值 所以我们继续简化
caculateFun(a: 10, b: 2) { (x, y) in
return x * y
} //20
//最后我们还可以忽略指定参数名,使用默认参数名$0(如果函数接收多个参数,使用$K作为第K-1个参数,如$0,$1,$2......)
caculateFun(a: 10, b: 40){
$0 * $1
}
//所以我们的例子在Swift中的闭包最简形式就变成
caculateFun(a: 10, b: 40){ $0 * $1}
二、高阶函数
接下来我们再看下高阶函数Map,Filter,Reduce的使用,可以极大的简化我们的代码
Map
map用于将每个数组元素通过某个方法进行转换。
//有一个数组 我们想对每一个数组中的元素都+1
let array = [1,2,3,4,5,6]
//之前我们一般创建一个空数组 遍历源数组进行操作
var resultArray = [Int]()
for x in array {
resultArray.append(x + 1)
}
//现在我们一行搞定
resultArray = array.map{$0 + 1}
Filter
filter用于选择数组元素中满足某种条件的元素。
//我们要筛选大于零的
let numbers = [2, -5, 9, 7, -2, 5, 3, 1, 0, -3, 8]
//传统
var result = [Int]()
for x in numbers {
result.append(x + 1)
}
//使用Filter
result = numbers.filter{$0 > 0}
Reduce
reduce方法把数组元素组合计算为一个值。
//我们要求和
let numbers = [2, -5, 9, 7, -2, 5, 3, 1, 0, -3, 8]
//传统
var result = 0
for x in numbers {
result += x
}
//使用reduce
result = numbers.reduce(0,{$0+$1})