最近用到了 swift 中的閉包,突然發(fā)現(xiàn)和之前用到的有些小改變,所以準(zhǔn)備寫個(gè)筆記記錄一下,希望和想了解閉包的小伙伴一起學(xué)習(xí)
閉包的理解
- 閉包是一個(gè)代碼塊(類似 OC 中 block)
2.使用場(chǎng)景:異步回調(diào).控制器回調(diào).自定義視圖回調(diào)(和 block 是一樣的)
閉包的基本寫法
1.沒有參數(shù)沒有返回值
a.函數(shù)是閉包的特殊寫法,也就是說我們平時(shí)用的函數(shù)也是閉包
b.閉包的標(biāo)準(zhǔn)表達(dá)式{()->() in 代碼}
函數(shù)名 = {形參列表 -> 返回值類型 in 實(shí)現(xiàn)代碼}
c.如果沒有參數(shù)沒有返回值()->()可以省略,連 in 都一起省略
let b1 = {
print("hello_我是最簡(jiǎn)單的閉包")
}
b1()
2.有參數(shù)沒有返回值
a.閉包中,參數(shù),返回值,實(shí)現(xiàn)代碼都是寫在{}中
b. 需要用關(guān)鍵字 in 來分割定義和實(shí)現(xiàn) 代碼
let b2 = {(x: Int) -> () in
//實(shí)現(xiàn)代碼
print(x)
}
b2(100)
- 帶參數(shù)帶返回值的閉包
a.參數(shù) x,y的類型是 Int 返回值的類型也是 Int 用 in 來分割
let b3 = {(x: Int,y: Int) -> Int in
return x+y
}
print(b3(100,400))
//輸出500
閉包的應(yīng)用
- 通過閉包的回調(diào)傳遞參數(shù)
定義一個(gè)函數(shù),將閉包當(dāng)做函數(shù)的一個(gè)參數(shù)
func loaData(completion:@escaping (_:[String])->()) -> () {
//將任務(wù)加載到隊(duì)列,指定執(zhí)行任務(wù)的函數(shù)
//翻譯: 隊(duì)列調(diào)度任務(wù)(block/ 閉包), 以同步異步的方式進(jìn)行
//使用閉包的場(chǎng)景和 block 一樣
DispatchQueue.global().async {
print("耗時(shí)操作\(Thread.current)")
//休眠
Thread.sleep(forTimeInterval: 1.0)
//獲得結(jié)果
let json = ["陜西","上海","出大事了"]
DispatchQueue.main.async(execute: {
print("主線程更新 UI\(Thread.current)")
//使用閉包進(jìn)行回調(diào) -> 執(zhí)行閉包(通過參數(shù)傳遞的)
completion(json)
})
}
}
注:1.當(dāng)函數(shù)中的參數(shù)是閉包函數(shù)的時(shí)候, 閉包中參數(shù)的名字可以忽略不寫,或者用"_"來代替 這個(gè)是 swift4.0 做出的改變,
2 ,關(guān)鍵字escaping 逃逸閉包: 如果這個(gè)閉包是在函數(shù)執(zhí)行完后才被調(diào)用畏铆,調(diào)用的地方超過了這個(gè)函數(shù)的范圍酪穿,這種閉包叫逃逸閉包章鲤。3.0以后的閉包默認(rèn)是非逃逸的,以上代碼中在主線程中調(diào)用閉包,已經(jīng)超出了函數(shù)的范圍,要加上關(guān)鍵字 escaping
- 閉包的調(diào)用
在viewDidLoad 中調(diào)用
override func viewDidLoad() {
super.viewDidLoad()
a.正常的寫法
loaData(completion: {(result)->() in
print("獲取的數(shù)據(jù)\(result)")
})
b. 該函數(shù)引出一個(gè)定義,尾隨閉包
如果函數(shù)的最后一個(gè)參數(shù)是閉包,那么函數(shù)的參數(shù)可以提前結(jié)束(其他參數(shù)可以不寫), 最后一個(gè)參數(shù)使用{}包裝閉包的代碼.所以直接寫成
loaData { (result) in
print("獲取的數(shù)據(jù)\(result)")
}
}
閉包還有很多知識(shí)點(diǎn)再深入學(xué)習(xí), 有時(shí)間還會(huì)繼續(xù)更新...