delegate 和 block 避免循环引用

循环引用

循环引用,顾名思义就是多个对象之间相互引用,ARC(Automatic Reference Counting)机制下很容易产生循环引用,在这种情况下会导致内存泄露,当然这是我们作为开发者所不能允许的,所以我们要避免这种情况的发生。

循环引用会在我们的开发中经常涉及到,今天我们主要看下delegateblock中什么样的情况不会出现循环引用,又有什么样的情况会出现循环引用以及如何解决。

代码说明

Objective-CSwift写的两个小Demo来说明。后面会给出下载地址。

Demo是以一个UINavigationController控制界面的跳转,界面的流程是:ViewController->OneViewController->TwoViewController或者ThreeViewController

App Flow

delegate

1 分别在OneViewControllerTwoViewController中重写了delloc方法,在他们的对象销毁时会在控制台打印出Log:

- (void)dealloc {NSLog(@"OneViewController dealloc");}

- (void)dealloc {NSLog(@"TwoViewController dealloc");}

2OneViewController中声明一个全局变量twoController:

@interfaceOneViewController(){    TwoViewController *twoController;}

3TwoViewController中定义了一个delegate:

@property(strong,nonatomic)id delegate;

4OneViewControllerDelegate Circular Ref按钮链接的函数是:

- (IBAction)delegateCircularRefButtonClick {    twoController = [TwoViewController new];    twoController.delegate =self;    [self.navigationController pushViewController:twoController animated:YES];}

点击Delegate Circular Ref按钮Push到TwoViewController后,然后依次点击返回按钮Pop到ViewController,会发现控制台并没有打印出任何Log。这说明OneViewControllerTwoViewController的对象并没有释放内存,因为这里出现了循环引用。

self -> twoController -> delegate -> self

这就是造成对象不会释放的原因,对象直接互相引用着,导致了内存泄露。我们接着往下看:

5OneViewControllerDelegate No Circular Ref按钮链接的函数是:

- (IBAction)delegateNoCircularRefButtonClick {`    TwoViewController *controller = [TwoViewController new];    controller.delegate =self;    [self.navigationController pushViewController:controller animated:YES];}

点击Delegate No Circular Ref按钮Push到TwoViewController后,然后点击返回按钮Pop到OneViewController,会发现控制台打印出Log:

TwoViewController dealloc

再次点击返回按钮Pop到ViewController,会发现控制台打印出Log:

OneViewController dealloc

这说明OneViewControllerTwoViewController的对象销毁释放内存,这里没有出现循环引用。

controller -> delegate -> self

两段代码比较,很容易发现哪里出现了循环引用,第一段代码中self持有了twoController,造成了一个引用环。

重要说明:不要因为第二种情况不会出现循环引用就可以在工程中对delegate使用strong属性,还是要使用weak属性!正确的代码是:

@property(weak,nonatomic)id delegate;

block

1ThreeViewController中重写了delloc方法,在对象销毁时会在控制台打印出Log:

- (void)dealloc {NSLog(@"ThreeViewController dealloc");}

2OneViewController中声明一个全局变量threeController:

@interfaceOneViewController(){    ThreeViewController *threeController;}

OneViewController中定义了一个实例变量:

@property(strong,nonatomic)NSString*name;

OneViewControllerviewDidLoad中对实例变量赋值:

self.name =@"Paolo Maldini";`

3ThreeViewController中定义了一个block:

typedefvoid(^BlockTest) ();@interfaceThreeViewController:UIViewController@property(copy,nonatomic) BlockTest block;@end

4ThreeViewControllerviewDidDisappear中执行block:

_block();

5OneViewControllerBlock Circular Ref按钮链接的函数是:

- (IBAction)blockCircularRefButtonClick {    threeController = [ThreeViewController new];    threeController.block = ^() {NSLog(@"Hello, %@!",self.name);    };    [self.navigationController pushViewController:threeController animated:YES];}

Xcode 已经提示 “Capturing 'self' strongly in this block is likely to lead to a retain cycle”

点击Block Circular Ref按钮Push到ThreeViewController后,然后依次点击返回按钮Pop到ViewController,会发现控制台只打印出block里的内容

Hello, Paolo Maldini!

这说明OneViewControllerThreeViewController的对象并没有释放内存,因为这里出现了循环引用。

self -> threeController -> block -> self

这就是造成对象不会释放的原因,对象直接互相引用着,导致了内存泄露。我们接着往下看:

6OneViewControllerBlock No Circular Ref按钮链接的函数是:

- (IBAction)blockNoCircularRefButtonClick {    ThreeViewController *controller = [ThreeViewController new];    controller.block = ^() {NSLog(@"Hello, %@!",self.name);    };    [self.navigationController pushViewController:controller animated:YES];}

点击Block No Circular Ref按钮Push到ThreeViewController后,然后点击返回按钮Pop到OneViewController,会发现控制台打印出Log:

Hello, Paolo Maldini!

ThreeViewController dealloc

再次点击返回按钮Pop到ViewController,会发现控制台打印出Log:

OneViewController dealloc

这说明OneViewControllerThreeViewController的对象销毁释放内存,这里没有出现循环引用。

controller -> block -> self

两段代码比较,很容易发现哪里出现了循环引用,第一段代码中self持有了threeController,造成了一个引用环。修改方法,用 *__weak *声明一个弱引用对象:

- (IBAction)blockCircularRefButtonClick {    threeController = [ThreeViewController new];    __weakOneViewController *weakSelf =self;    threeController.block = ^() {NSLog(@"Hello, %@!", weakSelf.name);    };    [self.navigationController pushViewController:threeController animated:YES];}

作者:iLB

链接:https://www.jianshu.com/p/22f27458d25a

來源:简书

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 不经作者同意,禁止转载 前言 最近在面试的过程中发现了一些问题,若干面试者对循环引用一知半解,知道有循环引用,但再...
    iLB阅读 1,929评论 1 11
  • 貌似 。。。。。理解有错误。。。。。。。。。。了 前言:闲着太久了,最近补血,此刻来兴致写点东西(1)说明一下内存...
    json_jie阅读 2,055评论 2 2
  • iOS网络架构讨论梳理整理中。。。 其实如果没有APIManager这一层是没法使用delegate的,毕竟多个单...
    yhtang阅读 5,257评论 1 23
  • http://pandara.xyz/2016/04/02/weakify_strongify/ 什么是循环引用 ...
    MSG猿阅读 917评论 0 3
  • 永嘉五年,十二月 入冬了,阿菟的腿伤也好的差不多了,郗子房还寄信给远在终南山跟随杨羲修习道术的方回讨要了生肌去疤的...
    复明的瞎子阅读 514评论 0 0