闭包练习

闭包表达式:匿名闭包

一、闭包引入

普通函数写法

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、由一对{}开始和结束;

2、in关键字把闭包分成两部分:参数与返回值、闭包体


第二部分:闭包缩写

需求:把java成绩放到了一个数组里 定义一个函数 把大于某个阈值的成绩返回

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:省略参数类型和括号(自动推断出参数类型是Int)

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


省略3:单行表达式闭包可以省略 return 关键字

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


省略4:省略参数列表定义,用$0,$1等等指代参数,同时省略in

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


第三部分:尾随闭包

使用情景:当闭包表达式作为最后一个参数传递给函数时 可以单独提出来

问题1:尾随指的是跟在别人后面 闭包到底跟在谁后面?-函数后面

问题2:用尾随闭包的好处?提升代码的可读性

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

    printFun(info)

}


//普通调用方式

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


//使用尾随闭包进行调用

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


值捕获

值捕获:也就是空手套白狼,把别人的变量或常量拿过来用

从案例中可以看出,incrementer内嵌函数自己没有定义任何的变量或常量,但可以捕获上下文中的常量和变量,拿过来自己使用。


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()) //输出结果为20


let c = makeIncrementer(forIncrement:10)

print(c())


逃逸闭包 逃到函数外使用


需求:闭包作为一个参数传递一个函数 但是这个闭包不立马使用 先把这个闭包存起来 过会再用

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

var x = 10

//方案一:定义一个函数,接受一个普通闭包为参数

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

    recv = closure //此段代码报错,原因是普通闭包作为参数 会在函数结束之后被销毁 无法在函数外使用。

}

test1{

    x=100

}

recv()


// 方案二:逃逸闭包

/*

逃逸闭包特点如下:

1、可以在函数结束后使用;

2、寿命长 逃逸闭包声明周期长于函数 只要它的引用被其他对象持有 就不会随着函数结束而释放掉

3、通过@escaping 指定一个闭包是逃逸闭包

*/

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

    recv = closure

}

test2{

    x = 200

}

recv()

print(x)

逃逸闭包在网络访问中会经常用到


思考1:demo是不是一个闭包

let demo = {print("hello")}

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


思考2:写一个闭包表达式,实现两数相加

let sum = {

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

      return num1+num2

}

print(sum(3,4))

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容