闭包与block-捕获变量差异

本文对比了Swift闭包与Objective-C (OC) 中block的变量捕获机制,通过具体示例展示了二者在变量引用行为上的不同之处,并介绍了如何在Swift中使用捕获列表来达到与OC相同的效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

  起初在学习使用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
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值