2020.03.31閉包作業(yè)

一、閉包引入

普通函數(shù)寫法


func square(num:Int)->Int{

return num*num

}

print(square(num:3))

閉包寫法


let fun1 = {

(num:Int)->Int in

return num*num

}

print(type(of:fun1))//fun1類型:(Int) -> Int

print(fun1(4))

閉包表達式語法

1、由一對{}開始和結(jié)束;

2、in關鍵字把閉包分成兩部分:參數(shù)與返回值、閉包體

思考1:demo是不是一個閉包


let demo = {print("hello")}

print(type(of:demo))// () -> ()

思考2:寫一個閉包表達式,實現(xiàn)兩數(shù)相加


let sum = {

(num1:Int,num2:Int)->Int in

return num1+num2

}

print(sum(3,4))

二、閉包縮寫

需求:把我們的java成績放到了一個數(shù)組里,定義一個函數(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:省略->返回類型(自動推斷出返回值是一個Bool)


print(getScore(score:score,con:{(s:Int) in return s>40}))

省略2:省略參數(shù)類型和括號(自動推斷出參數(shù)類型是Int)


print(getScore(score:score,con:{s in return s>40}))

省略3:單行表達式閉包可以省略 return 關鍵字來


print(getScore(score:score,con:{s in s>40}))

省略4:省略參數(shù)列表定義,用0,1等等指代參數(shù),同時省略in


print(getScore(score:score,con:{$0>40}))

三、尾隨閉包

使用情景:當閉包表達式作為最后一個參數(shù)傳遞給函數(shù)時,可以單獨提出來

問題1:尾隨指的是跟在別人后面 閉包到底跟在誰后面?-函數(shù)后面

問題2:用尾隨閉包的好處?提升代碼的可讀性


func printInfo(info:String,printFun:(String)->Void){

printFun(info)

}

//普通調(diào)用方式

printInfo(info:"hello world",printFun:{s in print(s+"~~~")})

//使用尾隨閉包進行調(diào)用

printInfo(info:"hello world"){s in print(s+"~~~")}

值捕獲

值捕獲:也就是空手套白狼,把別人的變量或常量拿過來用

從案例中可以看出,incrementer內(nèi)嵌函數(shù)自己沒有定義任何的變量或常量,但可以捕獲上下文中的常量和變量,拿過來自己使用。


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())

逃逸閉包

閉包要出逃,出逃到哪里呢?逃到函數(shù)外使用!牛掰!

需求:閉包作為一個參數(shù)傳遞一個函數(shù),但是這個閉包我不立馬使用,先把這個閉包存起來,過會再用


var recv:()->Void = {print("")}

var x = 10

//方案一:定義一個函數(shù),接受一個普通閉包為參數(shù)

func test1(closure:()->Void){

recv = closure //此段代碼報錯,原因是普通閉包作為參數(shù),會在函數(shù)結(jié)束之后被銷毀,無法在函數(shù)外使用。

}

test1{

x=100

}

recv()

// 方案二:逃逸閉包

/*

逃逸閉包特點如下:

1、可以在函數(shù)結(jié)束后使用;

2、壽命長!逃逸閉包聲明周期長于函數(shù),只要它的引用被其他對象持有,就不會隨著函數(shù)結(jié)束而釋放掉

3、通過@escaping 指定一個閉包是逃逸閉包

*/

func test2(closure:@escaping ()->Void){

recv = closure

}

test2{

x = 200

}

recv()

print(x)

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容