swift学习日志—— 闭包(二)

swift学习日志—— 闭包(二),第1张

概述非逃逸闭包 当一个闭包作为参数传到一个函数中,但是这个闭包在函数返回之后才被执行,我们称该闭包从函数中逃逸。当你定义接受闭包作为参数的函数时,你可以在参数名之前标注@noescape,用来指明这个闭包是不允许“逃逸”出这个函数的。将闭包标注@noescape能使编译器知道这个闭包的生命周期。 一种能使闭包“逃逸”出函数的方法是,将这个闭包保存在一个函数外部定义的变量中。举个例子,很多启动异步 *** 作的 非逃逸闭包 当一个闭包作为参数传到一个函数中,但是这个闭包在函数返回之后才被执行,我们称该闭包从函数中逃逸。当你定义接受闭包作为参数的函数时,你可以在参数名之前标注@noescape,用来指明这个闭包是不允许“逃逸”出这个函数的。将闭包标注能使编译器知道这个闭包的生命周期。
一种能使闭包“逃逸”出函数的方法是,将这个闭包保存在一个函数外部定义的变量中。举个例子,很多启动异步 *** 作的函数接受一个闭包参数作为 completion handler。这类函数会在异步 *** 作开始之后立刻返回,但是闭包直到异步 *** 作结束后才会被调用。在这种情况下,闭包需要“逃逸”出函数,因为闭包需要在函数返回之后被调用。例如:
var completionHandlers: [() -> VoID] = []func someFunctionWithEscaPingClosure(completionHandler: () -> VoID) {    completionHandlers.append(completionHandler)}

someFunctionWithEscaPingClosure(_:) 函数接受一个闭包作为参数,该闭包被添加到一个函数外定义的数组中。如果你试图将这个参数标注为 @noescape ,你将会获得一个编译错误。
将闭包标注为使你能在闭包中隐式地引用self
class SomeClass {    var x = 10    func doSomething() {        someFunctionWithEscaPingClosure { self.x = 100 }        someFunctionWithNoescapeClosure { x = 200 }    }}let instance = SomeClass()instance.doSomething()print(instance.x)// prints "200"completionHandlers.first?()print(instance.x)// prints "100"

也就是说非逃逸闭包和逃逸闭包讲的不是执行先后顺序,非逃逸是指你那个completionHandler闭包不能在函数外单独调用,只能在函数内部调用,函数调用完成后,那个闭包也就结束了,而逃逸闭包,你看官方给的例子,将闭包加入了数组,通过在外部访问数组元素,从而达到了使用那个闭包的效果。换句话说:如果闭包需要赋值给函数外部变量或常量的话,就不能使用非逃逸闭包了。 它其实是为了告诉函数调用者,表示这个闭包参数会在什么时候执行,也就是同步还是异步执行。同步就是在函数返回前就调用闭包,异步就是我先hold住这个闭包,然后函数立刻返回,等需要的时候再调用这个闭包。类似oc中的block参数,这个种逃逸闭包在网络请求中最常用了。
参考自The Swift Programming Language 官方文档 转载请附出处 转载jaccty的博客 总结

以上是内存溢出为你收集整理的swift学习日志—— 闭包(二)全部内容,希望文章能够帮你解决swift学习日志—— 闭包(二)所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1081462.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-27
下一篇 2022-05-27

发表评论

登录后才能评论

评论列表(0条)