8.swift3.0以上的(逃逸闭包@escaping)和(非逃逸闭包@noescape 编译器默认)(面试点:swift 1.0和swift2.0中,闭包默认是逃逸的,swift3.0以上默认...

闭包只有在函数中做参数时才会区分逃逸闭包和非逃逸闭包。
Swift 3.0之后,传递闭包到函数中的时候,系统会默认为非逃逸闭包类型(NonescapingClosures)@noescaping,逃逸闭包在闭包前要添加@escaping关键字。

闭包的发展历史

在Swift1.0和2.0中, 闭包参数默认是逃逸的, 如果你知道你的闭包参数是不会逃逸函数体的, 你可以用@escaping关键字修饰闭包参数.

在Swift3.0中, 变成了另外一种情况: 闭包参数默认是非逃逸的, 如果你的闭包试图逃逸函数体, 那么你要用@non-escaping来修饰闭包参数.

如果闭包是非逃逸的, 那么久存在一些潜在的优势, 因为闭包无法逃逸函数体, 那么编译器就可以优化闭包的存储和调用

在Swift3.0以上,函数的参数为闭包,默认为非逃逸闭包(@noescape) 在Swift2.0 或者 Swift1.0以下,没有关键字@noescape , 在Swift3.0 以上

写逃逸闭包就是

func someMethod(closure: @escaping () -> Void) {
  // secret stuff
}

从生命周期看两者区别:

  • 非逃逸闭包的生命周期与函数相同:
1,把闭包作为参数传给函数;

2,函数中调用闭包;

3,退出函数。结束
  • 逃逸闭包的生命周期:
1,闭包作为参数传递给函数;

2,退出函数;

3,闭包被调用,闭包生命周期结束

即逃逸闭包的生命周期长于函数,函数退出的时候,逃逸闭包的引用仍被其他对象持有,不会在函数结束时释放

例如:

非逃逸闭包:

class ToolClass:NSObject{
    func test(testBlock:(String)->()) {//1
        testBlock("非逃逸闭包")//2
    }//3
}

class ViewController: UIViewController {
    var tool:ToolClass = ToolClass.init()
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        tool.test { (str) in
            print(str)
        }
    }
    
    
}

代码执行顺序(1),(2),(3)
当传递闭包参数给函数test时,要注意ViewController中的属性tool,虽然闭包会捕获self,但是由于默认闭包参数是非逃逸型,这里可以省略self,编译器已经知道这里不会有循环引用的潜在风险。

逃逸闭包:

class ToolClass:NSObject{
    func test2(testBlock2:@escaping(String)->()) {//1
        DispatchQueue.global().async {
            DispatchQueue.main.async {
                testBlock("逃逸闭包")//2
            }
        }
    }//3
}
class ViewController: UIViewController {
    var tool:ToolClass = ToolClass.init()
    override func viewDidLoad() {
        super.viewDidLoad()
        tool.test2 { (str2) in
            print(str2)
        }
    }
}

代码执行顺序:(1),(3),(2)

当传递闭包参数给函数test2时,要注意ViewController中的属性tool,这里闭包函数的生命周期在函数结束后结束,tool前面省略的self 就有必要做特殊处理,防止造成循环引用weak var weakSelf = self。逃逸闭包前面添加@escaping关键字,这里闭包的生命周期不可预知。

经常使用逃逸闭包的2个场景:

  • 1.异步调用: 如果需要调度队列中异步调用闭包,比如网络请求成功的回调和失败的回调,这个队列会持有闭包的引用,至于什么时候调用闭包,或闭包什么时候运行结束都是不确定,上边的例子。

  • 2.存储: 需要存储闭包作为属性,全局变量或其他类型做稍后使用,例如

let kscreenWidth = UIScreen.main.bounds.size.width
let kscreenHeight = UIScreen.main.bounds.size.height

@objcMembers class AdvertiseView: UIView {
    private var dismisBlock: (() -> Void)?
    private var downBlock: (() -> Void)?
    private var completion: (() -> Void)?
    init(frame: CGRect, dismis: @escaping () -> Void, down: @escaping () -> Void, completion: @escaping () -> Void) {
        super.init(frame: frame)
        dismisBlock = dismis
        downBlock = down
        self.completion = completion
    }
    func test(){
        if (self.completion != nil) {
            self.completion!()
        }
    }
}

swift 逃逸闭包和非逃逸闭包的区别

swift 逃逸闭包和非逃逸闭包的区别
class NetworkManger {
    
    func getUserInfo(phone: String?, success:@escaping (() -> Void), failure: ((_ errorMessage: String) -> Void)) {
        print("函数开始执行")
        guard let _ = phone else {
            print("执行了failure闭包")
            failure("电话号码不能为空")
            return
        }
        //用来模拟网络请求
        let dataTask = URLSession.shared.dataTask(with: URL.init(string: "www.baidu.com")!) { (data, responmse, nil) in
            print("执行了success闭包")
            success()
        }
        dataTask.resume()
        print("函数执行结束")
    }
}

let netManger = NetworkManger()
netManger.getUserInfo(phone: "123456", success: {
    print("刷新你的界面")
}) { (errorMessage) in
    print(errorMessage)
}

逃逸闭包:闭包做为函数的参数传递时,在函数体结束后被调用,我们就说这个闭包逃离了这个函数体的作用域,这个闭包是逃逸型的闭包,使用@escaping来标注。

非逃逸型的闭包:在函数体结束前被调用,闭包是非逃逸型的闭包。

说明:
failure会在phone号码为空的时候触发,这个时候函数体未执行完毕,是非逃逸闭包。
success闭包是在一个异步线程中,当异步线程执行完毕后,我们才调用了success闭包,此时函数体已经执行完毕了,是逃逸闭包。

逃逸闭包应用场景:

1.异步调用: 如果需要调度队列中异步调用闭包, 这个队列会持有闭包的引用,至于什么时候调用闭包,或闭包什么时候运行结束都是不可预知的。
2.存储: 需要存储闭包作为属性,全局变量或其他类型做稍后使用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容