闭包与block-捕获变量差异

Posted WoodBear009

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了闭包与block-捕获变量差异相关的知识,希望对你有一定的参考价值。

  起初在学习使用swift时,对闭包的变量捕获部分并没有认真的去了解,想当然的认为应给和OC中的block捕获规则一样,结果最近在使用中产生了错误,回过头来再去了解时,发现了二者的捕获规则有着很大的差异,这里只说现象,提醒大家在使用中多加注意,至于原因,有兴趣的可以分别去具体了解一下闭包与block的变量捕获机制.

  先看OC代码

    TestModel *test = [[TestModel alloc] init];
    test.name = @"1";

    dispatch_after(2, dispatch_get_global_queue(0, 0), ^
        NSLog(@"%@",test.name);
    );
    test.name = @"2";
结果:2

    TestModel *test = [[TestModel alloc] init];
    test.name = @"1";

    dispatch_after(2, dispatch_get_global_queue(0, 0), ^
        NSLog(@"%@",test.name);
    );

    test = [[TestModel alloc] init];
    test.name = @"2";
结果:1

  熟悉OC的同学应该可以理解上面两个例子的结果输出

再看看swift

        var model = TestModel()
        model.name = "1"
        DispatchQueue.global().asyncAfter(deadline: .now() + 2) 
            print(model.name)
        
        model.name = "2"
结果:2
重点来了:

     .  var model = TestModel()
        model.name = "1"
        DispatchQueue.global().asyncAfter(deadline: .now() + 2) 
            print(model.name)
        
        model = TestModel()
        model.name = "2"
结果:2!这里从现象上与OC的block出现了完全不同的结果,闭包在使用model时,使用的是最新指向的对象。

如何让其表现的和OC一样呢?使用捕获列表,如下

        var model = TestModel()
        model.name = "1"
        DispatchQueue.global().asyncAfter(deadline: .now() + 2)  [model] in
            print(model.name)
        
        model = TestModel()
        model.name = "2"
结果:1

以上是关于闭包与block-捕获变量差异的主要内容,如果未能解决你的问题,请参考以下文章

Swift 闭包 与 OC block区别

Swift 中的闭包与 C 和 Objective-C中的 blocks 以及其它一些编程语言中的 lambdas 比較类似。

Swift之深入解析闭包Closures的使用和捕获变量的原理

Swift:闭包(Closures)

Swift 学习- 08 -- 闭包

swift闭包新手详解(新手必看)