iOS 控件详解

iOS Block

2018-10-25  本文已影响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

上一篇下一篇

猜你喜欢

热点阅读