iOS开发杂货铺面试传值相关

block解决循环引用详细解释

2016-04-21  本文已影响1855人  浩杰ee

一、产生原因

网上大部分帖子都表述为:"block里面引用了self导致循环引用",会让大家理解为要显式地出现"self"就会引起循环引用。其实这种说法是很不严谨的。

下面用代码来说明:
*1 声明几个block和一个属性:

@interface ViewController (){
void(^myBlock1)(void);
void(^myBlock2)(void);
void(^myBlock3)(void);
 }
@property (nonatomic,copy) NSString *person;
@end

*2 使用weakSelf 不会 引起循环引用(以下是最常用但是不提倡方法,后面会提到原因,继续看下去哦):

__weak typeof(self) weakSelf = self;

NSLog(@"init--> value:%@,address=%p,self=%p",self.person,self.person,self);
  myBlock1 = ^(void){
//这样不会造成循环引用
NSLog(@"execute1--> value:%@,address=%p,weakSelf=%p",weakSelf.person,weakSelf.person,weakSelf);
 };

*3 直接使用self,会循环引用:Xcode会给警告

myBlock2 = ^(void){
//这样造成循环引用
NSLog(@"execute2--> value:%@,address=%p,self=%p",self.person,self.person,self);
};

*4 要执行的方法抽取出来,也不会循环引用:

 myBlock3 = ^(void){
//这样也不会造成循环引用,已经抽取出要执行的方法
[weakSelf myBlock3Func];
 }; 
- (void)myBlock3Func{
NSLog(@"execute3--> value:%@,address=%p,self=%p",self.person,self.person,self);
 }

*5 block不是self的属性或者变量时,在block内使用self也不会循环引用:

 //block不是self的属性时,block内部使用self也不是循环引用
 Animal *animal = [[Animal alloc] init];
 animal.animalBlock = ^(void){
 NSLog(@"animal-->    value:%@,address=%p,self=%p",self.person,self.person,self);
};

所以说并不是在block中使用self必定会循环引用,要分情况处理,如果产生了循环引用如何解决呢:

二、解决循环引用

*1 为了方便使用,首先定义两个宏

#ifndef weakify
#define weakify(o) __typeof__(o) __weak o##__weak_ = o;
     #define strongify(o) __typeof__(o##__weak_) __strong o =    o##__weak_;
#endif

*2 使用如下代码解决循环引用

weakify(self);
success:^(AFHTTPRequestOperation *operation, id responseObject) {
    strongify(self);
    if (!self__weak_) return ;
   //...................
   }

3* 总结解释上述代码作用

三、 代码验证

*1 我们自定义一个类Text,在该类dealloc方法中加一行打印;

  -(void)dealloc{
 NSLog(@"dealloc %@被销毁了!!!!!!",[self class]);
}

*2 将Text作为另外一个类的属性

 @property(nonatomic,strong)Text *text;

*3 测试结果

 Text *text=[[Text alloc]init];
  self.text=text;

weakify(_text);

 //开启子线程      
      dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), ^{

        NSInteger count =0;

      strongify(_text);
     
        while( count<10) {

            count++;

            NSLog(@"---------%@---%ld",_text__weak_,(long)count);

            sleep(1);

        }

    });

 //3秒后将 text对象 销毁
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3*NSEC_PER_SEC)),dispatch_get_main_queue(), ^{
        
        self.text=nil;
        
    });

4* 第一次,不使用 strongify(_text);

屏幕快照 2016-04-21 下午9.46.30.png

5*第二次,使用 strongify(_text);


屏幕快照 2016-04-21 下午9.46.05.png

可以清楚的看到,添加了strongify(_text),系统就会等待block执行完成后再释放text对象,该text对象在block中的使用起到了保护作用。当block执行结束后会自动释放掉

上一篇下一篇

猜你喜欢

热点阅读