美文网首页
delegate 和 block 避免循环引用

delegate 和 block 避免循环引用

作者: YinLei | 来源:发表于2018-03-08 10:03 被阅读0次

循环引用

循环引用,顾名思义就是多个对象之间相互引用,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

來源:简书

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

相关文章

网友评论

      本文标题:delegate 和 block 避免循环引用

      本文链接:https://www.haomeiwen.com/subject/cavefftx.html