如何正确的使用Block

循环引用(原理)

1.给大家简单的概括一下循环引用出现的原理

A持有B  但是B又强引用A,而A在销毁的时候,就出现了所谓的循环引用

case1:

//ClassA:

@protocol ClssADelegate

- (void)fuck;

@end

@interface ClassA : UIViewController

@property (nonatomic, strong) id delegate;

@end

//ClassB:

@interface ClassB ()

@property (nonatomic, strong) ClassA *classA;

@end

@implementation ClassB

- (void)viewDidLoad {

    [super viewDidLoad]; 

    self.classA = [[ClassA alloc] init];  

    self.classA.delegate = self;

}

简单分析上面的例子:

B持有A属性,A的delegate强引用了A,这样就会导致循环引用

解决的最好办法就是

将@property (nonatomic, strong) id delegate;

改成:

@property (nonatomic, weak) id delegate;

case 2:

@interface ClassA ()

@property (nonatomic, copy) dispatch_block_t block;

@property (nonatomic, assign) NSInteger tem;

@end

@implementation ClassA

- (void)viewDidLoad {

    [super viewDidLoad];

    self.block = ^{

        self.tem = 1;

    };  

}


self持有block,而堆上的block又会持有self,所以会导致循环引用



修改方法:

 __weak typeof(self) weakSelf = self

    self.block = ^{

        weakSelf.tem = 1;

    };  


结论

如上delegate和block引起的循环引用的处理方式,有一个共同的特点,就是使用weak(弱引用)来打破环,使环消失了。所以,可以得出结论,我们可以通过使用将strong(强引用)用weak(弱引用)代替来解决循环引用。





进一步学习 深处探讨


//ClassB是一个UIViewController,假设从ClassA pushViewController将ClassB展示出来

@interface ClassB ()

@property (nonatomic, copy) dispatch_block_t block;

@property (nonatomic, strong) NSString *str;

@end

@implementation ClassB

- (void)dealloc {

}

- (void)viewDidLoad {

    [super viewDidLoad];

    self.str = @"111";

    __weak typeof(self) weakSelf = self;

    self.block = ^{

        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

            NSLog(@"%@", weakSelf.str);

        });

    };

    self.block();   

}

这里会有两种情况:

若从A push到B,10s之内没有pop回A的话,B中block会执行打印出来111。

若从A push到B,10s之内pop回A的话,B会立即执行dealloc,从而导致B中block打印出(null)。这种情况就是使用weakSelf的缺陷,可能会导致内存提前回收。

2、weakSelf和strongSelf

@interface ClassB ()

@property (nonatomic, copy) dispatch_block_t block;

@property (nonatomic, strong) NSString *str;

@end

@implementation ClassB

- (void)dealloc {

}

- (void)viewDidLoad {

    [super viewDidLoad];

    self.str = @"111";

    __weak typeof(self) weakSelf = self;

    self.block = ^{

        __strong typeof(self) strongSelf = weakSelf;

        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

            NSLog(@"%@", strongSelf.str);

        });

    };

    self.block();   

}

我们发现这样确实解决了问题,但是可能会有两个不理解的点。

这么做和直接用self有什么区别,为什么不会有循环引用:外部的weakSelf是为了打破环,从而使得没有循环引用,而内部的strongSelf仅仅是个局部变量,存在栈中,会在block执行结束后回收,不会再造成循环引用。

这么做和使用weakSelf有什么区别:唯一的区别就是多了一个strongSelf,而这里的strongSelf会使ClassB的对象引用计数+1,使得ClassB pop到A的时候,并不会执行dealloc,因为引用计数还不为0,strongSelf仍持有ClassB,而在block执行完,局部的strongSelf才会回收,此时ClassB dealloc。

这样做其实已经可以解决所有问题,但是强迫症的我们依然能找到它的缺陷:

block内部必须使用strongSelf,很麻烦,不如直接使用self简便。

很容易在block内部不小心使用了self,这样还是会引起循环引用,这种错误很难发觉。

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转:https://www.jianshu.com/p/51bb714051ea 谈到循环引用,不知道你能想到什么...
    高思阳阅读 4,567评论 0 0
  • 在讲block的循环引用问题之前,我们需要先了解一下iOS的内存管理机制和block的基本知识 iOS的内存管理机...
    爱吃鱼的老丑哥阅读 3,414评论 0 0
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 13,871评论 1 32
  • 1.设计模式是什么? 你知道哪些设计模式,并简要叙述? 设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类...
    司马DE晴空阅读 5,145评论 0 7
  • 第一篇:相识 今日有空想聊聊我的父亲母亲。 父亲母亲的相识怕是在现在这个年代来看都还是...
    玮玮萌语阅读 734评论 0 0

友情链接更多精彩内容