iOS Block

作者: 风冰武 | 来源:发表于2018-10-25 14:44 被阅读3次

    1:block使用局部变量

    在block内使用外部的局部变量时, 如果没有__block修改, 那么是会报错的: 图1

    此时在block内部的变量是只读的(readonly); 如果想要在block内部修改外部的局部变量, 有两种方法:

    (1)用__blcok修饰变量: 图2
    (2)用static修饰变量:
    图3

    2: __block的作用:

    (1)没有添加__block时: 图4
    图5

    由图5可以看出:
    在栈中的地址: 保持不变;
    在堆中的地址: 在block外保持不变, 在block内则是新建一个内存地址指向原来的变量, block作用域结束则销毁, 不影响原变量;

    (2)添加__block, 仅访问外部变量:
    图6
    图7

    由图7可以看出:
    在堆中的地址: 保持不变;
    在栈中的地址: 在block内新建一个内存地址之后, 就一直保持不变; block作用域结束也不会销毁;

    (3)添加__block, 改变外部变量的值:
    图8
    图9

    由图9可以看出:
    在堆中的地址: 在block内新建一个内存地址之后, 就保持不变;
    在栈中的地址: 在block内新建一个内存地址之后, 就保持不变;

    总结:

    3:在block内的强引用和弱引用的问题

    创建一个测试demo, 包含一个普通的person类

    .h文件
    
    #import <Foundation/Foundation.h>
    
    @interface XFPerson : NSObject
    //name
    @property (nonatomic, copy) NSString *name;
    //block
    @property (nonatomic, copy) void(^block)(void);
    
    @end
    

    另外, 为了检测person对象是否被释放, 在XFPerson.m内加上一段测试代码:

    #import "XFPerson.h"
    
    @implementation XFPerson
    
    /** 监测person是否释放内存 */
    - (void)dealloc {
        NSLog(@"%s", __func__);
    }
    
    @end
    

    接着来到viewController.m中, 引入person类的头文件, 实例化一个person对象. 观察以下的几种情况:

    3.1: block内部没有使用外部变量

    给name和block赋值

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
        
        person.block = ^{
            NSLog(@"block--block");
        };
        //调用 block 对象
        person.block();
    }
    输出:
    block--block
    -[XFPerson dealloc]
    
    结果: block内容被输出, person对象被正常释放
    
    

    3.2: block内部使用外部声明的强引用访问person对象: 在block内部访问person的name属性

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
    
       person.block = ^{
            NSLog(@"block--%@",person.name);//
        };
        //调用 block 对象
        person.block();
    }
    
    输出:block--xiaofan
    
    结果:
    (1)使用外部声明的强引用, 执行完毕后, 引用对象不会被销毁;
    
    (2)代码执行完毕后, block内容被输出, 但并没有发现 person 对象被正常释放, 此时会产生内存问题;
    
    (3)此时会有警告:Capturing 'person' strongly in this block is likely to lead to a retain cycle
    
    

    3.3: block内部使用外部声明的弱引用访问person对象: 在 block 内部使用外部声明的弱引用访问 person 的name对象

    - (void)viewDidLoad {
        [super viewDidLoad];
    
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
        
        __weak XFPerson *weakPerson = person;
        person.block = ^{
            NSLog(@"block--%@",weakPerson.name);
        };
        //调用 block 对象
        person.block();
    }
    输出:
    block-xiaofan
    -[XFPerson dealloc]
    
    结果:
    (1)使用外部声明的弱引用, 执行完毕, 引用对象被释放;
    
    

    可能你会说, 以后内部就使用外部声明的弱引用不就可以了, 但是, 看以下的情况:

    3.4: 延迟操作中使用弱引用

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
    
        __weak XFPerson *weakPerson = person;
        person.block = ^{
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                NSLog(@"block--%@",weakPerson.name);//弱引用的时候, 一旦block延迟操作, 就出问题了;
            });
        };
        //调用 block 对象
        person.block();
    }
    
    输出:-[XFPerson dealloc]
    3秒后
    block--(null)
    
    结果:
    (1)在打印block内容之前, person对象已经被释放了, 自然, name的属性值也是空了;
    
    

    3.5: block 内部有延迟操作的时候, 使用外部声明的强引用

    这种情况是针对上面那个测试的, 就是在延迟操作的时候, 直接使用外部声明的强引用

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
    
        person.block = ^{
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                NSLog(@"block--%@",person.name);
            });
        };
        //调用 block 对象
        person.block();
    }
    
    输出:
    block--xiaofan
    
    结果: 
    (1)这个时候, 确实可以访问到 person 对象的 name 属性值, 但是, 同样像最开始一样, block 会对 person 对象产生强引用, 代码执行完毕后 person 对象没有被正常释放, 导致内存问题;
    
    (2)有警告: Capturing 'person' strongly in this block is likely to lead to a retain cycle
    
    

    3.6: block 内部使用内部声明的强引用

    在 block 内部声明一个强引用指向外部声明的弱引用

    - (void)viewDidLoad {
        [super viewDidLoad];
        
        XFPerson *person = [XFPerson new];
        person.name = @"xiaofan";
    
        __weak XFPerson *weakPerson = person;
        person.block = ^{
            //强引用:
            XFPerson *strongPerson = weakPerson;
    
            dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
                NSLog(@"block--%@",strongPerson.name);
            });
        };
        //调用 block 对象
        person.block();
    }
    
    输出:
    block--xiaofan
    -[XFPerson dealloc]
    
    结果:
    person 的 name 属性值被打印出来, 而且, person 也被正常释放;
    
    

    参考:
    1: https://www.jianshu.com/p/ad1c366d6d11
    2: https://www.jianshu.com/p/1d8691e19ed9

    相关文章

      网友评论

        本文标题:iOS Block

        本文链接:https://www.haomeiwen.com/subject/hfeqtqtx.html