閉包:是自包含的功能代碼塊享潜,可以在代碼中使用或者用來作為參數(shù)傳值
Swift 中的閉包與 C 和 Objective-C 中的代碼塊以及其他一些編程語言中的匿名函數(shù)比較相似
全局函數(shù)和嵌套函數(shù)其實(shí)就是特殊的閉包
*全局函數(shù):有名字但不能捕獲任何值
*嵌套函數(shù):有名字珊膜,也能捕獲封閉函數(shù)內(nèi)的值
*閉包表達(dá)式:無名閉包堵幽,使用輕量級(jí)語法,可以根據(jù)上下文環(huán)境捕獲值
一骄瓣、閉包引入
普通函數(shù)寫法
func square(num:Int)->Int{
return numnum
}
print(square(num:3))
閉包寫法
let fun1 = {
(num:Int)->Int in
return numnum
}
print(type(of:fun1))//fun1類型:(Int) -> Int
print(fun1(4))
閉包表達(dá)式:是一種利用簡潔語法構(gòu)建內(nèi)聯(lián)閉包的方式求泰。 閉包表達(dá)式提供了一些語法優(yōu)化备畦,使得撰寫閉包變得簡單明了
1、由一對(duì){}開始和結(jié)束
2仁期、in關(guān)鍵字把閉包分成兩部分:參數(shù)與返回值桑驱、閉包體
二竭恬、閉包縮寫
func getScore(score:[Int],con:(Int)->Bool)->[Int]{
var newScore = Int
for item in score{
if con(item){
newScore.append(item)
}
}
return newScore
}
var score = [66,89,80,33,100]
print(getScore(score:score,con:{(s:Int)->Bool in return s>40}))
省略1:省略->返回類型(自動(dòng)推斷出返回值是一個(gè)Bool)
print(getScore(score:score,con:{(s:Int) in return s>40}))
省略2:省略參數(shù)類型和括號(hào)(自動(dòng)推斷出參數(shù)類型是Int)
print(getScore(score:score,con:{s in return s>40}))
省略3:單行表達(dá)式閉包可以省略 return 關(guān)鍵字來
print(getScore(score:score,con:{0,
0>40}))
三痊硕、尾隨閉包
尾隨閉包是一個(gè)書寫在函數(shù)括號(hào)之后的閉包表達(dá)式,函數(shù)支持將其作為最后一個(gè)參數(shù)調(diào)用
func printInfo(info:String,printFun:(String)->Void){
printFun(info)
}
//普通調(diào)用方式
printInfo(info:"hello world",printFun:{s in print(s+"~~~")})
//使用尾隨閉包進(jìn)行調(diào)用
printInfo(info:"hello world"){s in print(s+"~~~")}
值捕獲
閉包可以在其定義的上下文中捕獲常量或變量
即使定義這些常量和變量的原域已經(jīng)不存在押框,閉包仍然可以在閉包函數(shù)體內(nèi)引用和修改這些值
Swift最簡單的閉包形式是嵌套函數(shù)岔绸,也就是定義在其他函數(shù)的函數(shù)體內(nèi)的函數(shù)
嵌套函數(shù)可以捕獲其外部函數(shù)所有的參數(shù)以及定義的常量和變量
看這個(gè)例子:
func makeIncrementer(forIncrement amount: Int) -> () -> Int {
var runningTotal = 0
func incrementer() -> Int {
runningTotal += amount
return runningTotal
}
return incrementer
}
let a = makeIncrementer(forIncrement:10)
print(a())
閉包是引用類型
let b = a
print(b()) //輸出結(jié)果為20
let c = makeIncrementer(forIncrement:10)
print(c())
逃逸閉包
ar recv:()->Void = {print("")}
var x = 10
//方案一:定義一個(gè)函數(shù),接受一個(gè)普通閉包為參數(shù)
func test1(closure:()->Void){
recv = closure //此段代碼報(bào)錯(cuò)橡伞,原因是普通閉包作為參數(shù)盒揉,會(huì)在函數(shù)結(jié)束之后被銷毀,無法在函數(shù)外使用兑徘。
}
test1{
x=100
}
recv()
// 方案二:逃逸閉包
逃逸閉包特點(diǎn)如下:
1刚盈、可以在函數(shù)結(jié)束后使用;
2挂脑、壽命長藕漱!逃逸閉包聲明周期長于函數(shù),只要它的引用被其他對(duì)象持有崭闲,就不會(huì)隨著函數(shù)結(jié)束而釋放掉
3肋联、通過@escaping 指定一個(gè)閉包是逃逸閉包
func test2(closure:@escaping ()->Void){
recv = closure
}
test2{
x = 200
}
recv()
print(x)