今天我们继续探索block底层原理,为什么会造成循环引用,block的原理,block的结构等。
一. block的底层
1、block的循环引用
-(instancetype)init {
if (self = [super init]) {
void(^block)(void) = ^{
_name;
};
}
return self;
}
复制代码
这段代码会循环引用吗?答应是:会。为什么呢?
现在我们用clang来看看源码:
NSObject *objc = [NSObject new];
void (^block)(void) = ^ {
NSLog(@"%@",objc);
};
复制代码
clang -rewrite-objc main.m
后查看源码: 第一个问题得到答应:_name在block捕获中又变成了self。所以还是会循环引用。
那要如何解决循环引用呢?
self.block = ^{
weakself.name = @"lg";
};
复制代码
还有一个小细节,如果在block中使用延迟操作:
__weak typeof(self) weakself = self;
self.name = @"lg";
self.block = ^{
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSLog(@"%@",weakself.name);
});
};
**2022-06-19 17:41:31.215393+0800 Block的循环引用[13967:532378] -[LGViewController dealloc]**
**2022-06-19 17:41:32.146640+0800 Block的循环引用[13967:532378] (null)**
复制代码
最后打印结果为null,name被提前释放了?怎么办和我们想要的效果不一样啊。为解决这个问题使用了__strong
在block在强引一次。
self.name = @"lg";
__weak typeof(self) weakSelf = self;
self.block = ^{
__strong typeof(weakSelf) strongSelf = weakSelf;
NSLog(@"%p",&strongSelf);
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSLog(@"%@",strongSelf.name);
});
};
self.block();
**2022-06-19 17:45:21.113499+0800 Block的循环引用[14067:535530] 0x7ff7b0cdaf78**
**2022-06-19 17:45:24.113992+0800 Block的循环引用[14067:535530] lg**
**2022-06-19 17:45:24.114181+0800 Block的循环引用[14067:535530] -[LGViewController dealloc]**
复制代码
发现打印先打印了lg,在执行dealloc释放
。还有其他方法可以打破循环引用吗?当然有:使用__block
self.name = @"lg";
__block LGViewController *vc = self;
self.block = ^{
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
NSLog(@"%@",vc.name);
vc = nil; //手动释放
});
};
self.block();
**2022-06-19 17:55:33.469556+0800 Block的循环引用[14313:543068] lg**
**2022-06-19 17:55:33.469761+0800 Bloc