寫(xiě)在最前:這篇文章應(yīng)該是Block系列文章的最后一篇了,以我目前的水平來(lái)說(shuō)翘地,難免有錯(cuò)誤的地方。而且總體來(lái)說(shuō)癌幕,研究的比較淺衙耕。希望未來(lái)的自己重新審視Block的時(shí)候,會(huì)有更加深的認(rèn)識(shí)勺远。
下面是Block系列的所有文章:
- [iOS]Block系列探究一 - 初探
- [iOS]Block系列探究二 - 捕獲變量
- [iOS]Block系列探究三 - Block存儲(chǔ)域
- [iOS]Block系列探究四 - __block變量存儲(chǔ)域
- [iOS]Block系列探究五 - 截獲對(duì)象
- [iOS]Block系列探究六 - __block變量和對(duì)象
這次橙喘,我們來(lái)探究一下Block的循環(huán)引用,并且探討一下如何避免循環(huán)引用谚中。
一渴杆、什么是循環(huán)引用?
兩個(gè)對(duì)象直接或者間接的引用對(duì)方就是循環(huán)引用宪塔。
二、循環(huán)引用有什么問(wèn)題囊拜?
循環(huán)引用的兩個(gè)對(duì)象無(wú)法釋放某筐。
三、Block如何產(chǎn)生循環(huán)引用冠跷?
舉個(gè)栗子:
// interface聲明block南誊,被viewController實(shí)例對(duì)象強(qiáng)引用
@interface ViewController ()
@property (nonatomic, strong) void (^block)(void);
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// block引用了self
self.block = ^{
NSLog(@"self:%@", self);
};
self.block();
}
@end
上面的代碼,在退出viewController
的時(shí)候蜜托,block
因?yàn)楸?code>viewController強(qiáng)引用抄囚,引用計(jì)數(shù)不為0,無(wú)法銷毀橄务,導(dǎo)致了viewController
也同樣被block
強(qiáng)引用幔托,無(wú)法被銷毀。
四、怎么解決循環(huán)引用重挑?
下面我們來(lái)一步一步的研究應(yīng)該怎么避免循環(huán)使用嗓化。
4.1 __weak
根據(jù)[iOS]Block系列探究五 - 截獲對(duì)象我們發(fā)現(xiàn),堆Block
截獲__weak對(duì)象
不會(huì)強(qiáng)引用對(duì)象谬哀,所以我們把上面的栗子改寫(xiě)一下就能解決循環(huán)引用的問(wèn)題了刺覆,代碼如下:
// 弱引用self
__weak typeof(self) weakSelf = self;
self.block = ^{
// 捕獲__weak對(duì)象,不會(huì)強(qiáng)引用
NSLog(@"self:%@", weakSelf);
};
看上去我們解決了Block循環(huán)引用的問(wèn)題史煎,那么來(lái)看一下下面的情況谦屑。
4.2 __weak和__strong結(jié)合使用
先上代碼:
@interface ViewController ()
@property (nonatomic, strong) void (^block)(void);
@property (nonatomic, strong) NSMutableArray *arrM;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// 模擬數(shù)組引用計(jì)數(shù)為0時(shí)被銷毀的通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(releaseArrM) name:@"releaseArrM" object:nil];
[self method];
}
- (void)method {
// 模擬處理數(shù)據(jù)
self.arrM = [[NSMutableArray alloc] init];
[self.arrM addObject:self]; // 處理數(shù)據(jù)中引用了self
// 弱引用數(shù)據(jù)
__weak typeof(NSMutableArray *) weakArrM = self.arrM;
self.block = ^{
// 這里數(shù)組還沒(méi)有被釋放
[weakArrM addObject:@"哈哈"];
// 這里模擬數(shù)組被釋放
[[NSNotificationCenter defaultCenter] postNotificationName:@"releaseArrM" object:nil];
// 這里數(shù)組已經(jīng)被釋放了
NSLog(@"weakArrM的引用計(jì)數(shù)為:%@", @(CFGetRetainCount((__bridge CFTypeRef)(weakArrM))));
};
self.block();
}
- (void)releaseArrM {
// 這里演示數(shù)組引用計(jì)數(shù)為0時(shí)被銷毀
self.arrM = nil;
}
上面代碼中NSLog(@"weakArrM的引用計(jì)數(shù)為:%@", @(CFGetRetainCount((__bridge CFTypeRef)(weakArrM))));
處會(huì)崩潰,因?yàn)榇藭r(shí)數(shù)組已經(jīng)被釋放了篇梭。
那么怎么解決在block執(zhí)行中捕獲的對(duì)象被釋放引發(fā)的崩潰呢氢橙?
答案是在block內(nèi)使用__strong變量來(lái)強(qiáng)引用一下捕獲的__weak對(duì)象,保證在block塊返回之前捕獲的對(duì)象強(qiáng)引用計(jì)數(shù)不為0被釋放很洋。
改進(jìn)的代碼如下:
// 弱引用數(shù)據(jù)
__weak typeof(NSMutableArray *) weakArrM = self.arrM;
self.block = ^{
__strong typeof(NSMutableArray *) strongArrM = weakArrM;
// 這個(gè)時(shí)候strongArrm引用計(jì)數(shù)至少為1充蓝,不會(huì)被釋放。
...
};
至此喉磁,我們解決了block執(zhí)行的過(guò)程中捕獲的變量被釋放引起的問(wèn)題谓苟,那么block執(zhí)行前捕獲的變量已經(jīng)被釋放了怎么辦呢?我們還有下面兩種解決辦法协怒。
4.3 if條件鑒空
這個(gè)思路很簡(jiǎn)單涝焙,就是判斷block捕獲的對(duì)象是否為nil,不為nil才執(zhí)行之后的操作孕暇。
代碼如下:
// 弱引用數(shù)據(jù)
__weak typeof(NSMutableArray *) weakArrM = self.arrM;
self.block = ^{
__strong typeof(NSMutableArray *) strongArrM = weakArrM;
// 防止strongArrM為nil
if (strongArrM) {
// 這個(gè)時(shí)候strongArrM不為nil且引用計(jì)數(shù)至少為1仑撞,不會(huì)被釋放。
...
}
};
至少不會(huì)產(chǎn)生可能崩潰的情況了妖滔,但是業(yè)務(wù)代碼并沒(méi)有執(zhí)行隧哮,那么,我們?cè)趺幢WC既執(zhí)行業(yè)務(wù)代碼座舍,又解決循環(huán)引用呢沮翔?下面是一種使用場(chǎng)景很有限的解決方法。
4.4 __block配合置nil
我們結(jié)合代碼來(lái)說(shuō)明:
// 使用__block說(shuō)明符修飾
__block NSMutableArray *arrM = self.arrM;
self.block = ^{
// 這個(gè)時(shí)候強(qiáng)引用了arrM曲秉,arrM不可能為nil
[arrM addObject:@"哈哈"];
// 在block執(zhí)行的最后手動(dòng)把a(bǔ)rrM置nil采蚀,打破循環(huán)引用
arrM = nil;
};
使用__block說(shuō)明符
修飾被捕獲的對(duì)象,使對(duì)象在block中可以被修改承二,在執(zhí)行完業(yè)務(wù)代碼之后手動(dòng)把對(duì)象置為nil來(lái)打破循環(huán)引用榆鼠,但是這樣做有很大的限制:
- block必須執(zhí)行!:ヰ妆够!不執(zhí)行的話就循環(huán)引用了!!T鹁病T摹!
- 需要保證被捕獲的對(duì)象在block執(zhí)行之后不再使用T煮ΑL夥!因?yàn)閎lock執(zhí)行之后對(duì)象變?yōu)閚il了腰鬼。
和上面的方法類似的還有下面的方法嵌赠。
4.5 block執(zhí)行完將block置nil
和上面的方法類似,就是block執(zhí)行完手動(dòng)將block置為nil來(lái)打破循環(huán)引用熄赡,代碼如下:
self.block = ^{
// 這個(gè)時(shí)候強(qiáng)引用了self.arrM
[self.arrM addObject:@"哈哈"];
};
self.block();
// 手動(dòng)將block置為nil打破循環(huán)引用
self.block = nil;
綜上姜挺,我最推崇的是__weak+__strong+if鑒空來(lái)防止block的循環(huán)引用。但是希望block中的業(yè)務(wù)代碼一定要被執(zhí)行的話彼硫,其實(shí)有不少方法來(lái)解決循環(huán)引用的問(wèn)題炊豪,但是大致思想不外乎block執(zhí)行完手動(dòng)置nil打破循環(huán)引用。