一 、Block的内存泄露体现
block在copy时都会对block内部用到的对象进行强引用(ARC)或者retainCount增1(非ARC)。在ARC与非ARC环境下对block使用不当都会引起循环引用问题。 一般表现为: 1.某个类将block作为自己的属性变量,然后该类在block的方法体里面又使用了该类本身,简单说就是:
self.someBlock = ^(Type var){
[self dosomething];
self.otherVar = XXX;// 或者_otherVar = ...
};
block的这种循环引用会被编译器捕捉到并及时提醒。(这个还是很人性化的)
// Created by 58 on 15/6/3.
// Copyright (c) 2015年 58. All rights reserved.
//
#import"TsetBlock.h"
void (^blockTest)(NSString *str,NSString *st );
typedefvoid (^blockT)(NSString *str,NSString *st );
@interfaceTsetBlock ()@property (nonatomic)NSArray *testArr;
@property (nonatomic ,copy)blockT block;
@end
@implementation TsetBlock
- (id)init{
if (self = [superinit]) {
self.testArr =@[@"你",@"觉",@"的",@"会",@"怎",@"样"];
self.block = ^(NSString *name,NSString *str){
NSLog(@"arr:%@",self.testArr);
};
}
returnself;
}
@end
那么问题来了:
网上大部分帖子都表述为“block里面引用了self导致循环引用”,但事实真的是如此吗?我表示怀疑,其实这种说法是不严谨的,不一定要显式地出现”self”字眼才会引起循环引用。我们改一下代码,不通过属性self.arr去访问arr变量,而是通过实例变量_arr去访问,如下:
很明显了: 即使在你的block代码中没有显式地出现”self”,也会出现循环引用!只要你在block里用到了self所拥有的东西!但对于这种情况,目前我不知道该如何排除掉循环引用,因为我们无法通过加__weak声明或者__block声明去禁止block对self进行强引用或者强制增加引用计数。对于self.arr的情况,我们要分两种环境去解决:
1.arc:__weaktypeof(self) weakSelf=self; 其实 __weak someClass *weakSelf = self也是OK的!!!
2.MRC:解决方式与上述基本一致,只不过将__weak关键字换成__block即可,这样的意思是告诉block:孙子,咱们已经没有关系了(不要在内部对self进行retain了)!