1:block使用局部变量
在block内使用外部的局部变量时, 如果没有__block修改, 那么是会报错的:此时在block内部的变量是只读的(readonly); 如果想要在block内部修改外部的局部变量, 有两种方法:
(2)用static修饰变量:
2: __block的作用:
(1)没有添加__block时:
由图5可以看出:
在栈中的地址: 保持不变;
在堆中的地址: 在block外保持不变, 在block内则是新建一个内存地址指向原来的变量, block作用域结束则销毁, 不影响原变量;
(2)添加__block, 仅访问外部变量:
由图7可以看出:
在堆中的地址: 保持不变;
在栈中的地址: 在block内新建一个内存地址之后, 就一直保持不变; block作用域结束也不会销毁;
(3)添加__block, 改变外部变量的值:
由图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