block解决循环引用详细解释

*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 要执行的方法抽取出来,weakself 调用也不会循环引用:

 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* 总结解释上述代码作用

  • weakify(self);  创建一个指向self的弱引用
  • strongify(self);  当加上修饰符strong时,当别处把“self”释放掉,但调用该“self”的block如果仍然没有执行结束,那么系统就会等待block执行完成后再释放,对该“self”在block中的使用起到了保护作用。当block执行结束后会自动释放掉。
  • if (!self__weak_) return ; 进行判断,如果在执行strongify(self)之前“self已经被释放掉了,则此时self=nil,所以直接return即可”

三、 代码验证

*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);

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值