美文网首页iOS开发
iOS 关于 delegate 修饰词的错误认知

iOS 关于 delegate 修饰词的错误认知

作者: BlackStar暗星 | 来源:发表于2020-12-03 17:51 被阅读0次

不是所有的 delegate 都必须使用 weak 的,他是可以使用 strong 的

对于 delegate 很多情况确实需要我们使用 weak 来解决循环引用问题,但是有些情况使用 delegate 是不会出现 循环引用的。所以不要看见使用 strong 修饰 delegate 就认为错误。
案例

// A controller 中

-(void)pushTransition:(UIButton *)sender{
    BSATransitionController *aVC = [[BSATransitionController alloc]init];
    aVC.delegate = self;
    [self.navigationController pushViewController:aVC animated:YES];
}



// B controller 中
// ============ .h 中 ===========

@protocol BSATransitionControllerDelegate <NSObject>

@optional
-(void)refresh;

@end

@interface BSATransitionController : UIViewController

@property (nonatomic ,strong) id<BSATransitionControllerDelegate> delegate;

-(void)test;

@end


//============ .m 中 ===========

-(void)pop{
    [self.delegate refresh];
    [self.navigationController popViewControllerAnimated:YES];
}

虽然 delegate 使用了 strong ,使得 B 强持了 A ,但是 A 并没有持有 B ,没有形成一个完整的环,所以并不构成 循环引用,当返回的时候,B 页面 销毁 ,BA 的强持,仅仅 是 使 A引用计数器减一,并没有对 A 产生其他的影响。简单来说造不造成循环引用,先决条件是 A 引没引用持有 delegate 的那个对象。

为什么 tableview 的 delegate 和 source 用 weak ?因为我们在使用tableview的时候,几乎都会把 tableview 声明成属性,这就产生了我们所说的先决条件, 所以需要使用 weak 声明 delegate ,来打破循环引用



引发的问题:

如果 A 对 B 的持有 使用 weak 来声明,然后 仍然使用 strong 修饰 delegate 呢,不就同样构成了 A 没有持有 B ,但是 B 持有了A ,仍然不会循环引用吗 ?
通过验证确实如果,使用 weak 来声明 A 中的属性 B ,strong 来声明 B 中的 delegate 同样不会出现 循环引用(如果测试 你会发现 没法用 weak 声明 B ,因为需要如下操作才可以)

  BSATransitionController *aVC = [[BSATransitionController alloc]init];
  aVC.delegate = self;
  self.bvc = aVC;  
  [self.navigationController pushViewController:aVC animated:YES];

就是 说 ,我们不可以使用 self.bvc = [[BSATransitionController alloc]init]; 因为 bvc 是 weak 声明的,如果直接 self.bvc = [[BSATransitionController alloc]init]; 这么写,这行代码执行完,会立刻释放对象,self.bvc变成野指针,所以编译器是直接报错,不让我们这么用。但是如果是采用临时变量赋值的方法,那么在avc 没销毁的时候,即在函数作用域内,对象不会消失。

既然这样为什么不使用 weak 来声明B解决问题呢,而都是使用weak来声明delegate。 我的理解是:B 执行了 POP ,页面销毁 self.bvc 指针指向的地址被回收了,所以 self.bvc 变成了野指针,虽然在最开始使用 临时变量 的方法解决了系统报错的问题,但是等到 B 销毁,还是会出现 野指针的问题 。虽然我们可以在delegate回调的时候 可以这么写 self.bvc = nil ,但是毕竟不如 使用 weak 声明 delegate 来的方便。

验证 使用临时变量 给 bvc 赋值的方式,跳转后,B controller 释放,看self.bvc 是否会是野指针

-(void)pushTransition:(UIButton *)sender{
    
    dispatch_queue_t queue = dispatch_queue_create("quque", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), queue, ^{
        NSLog(@"%@==%p == %p",self.bvc,self.bvc ,&(self->_bvc));
    });
    
    BSATransitionController *aVC = [[BSATransitionController alloc]init];
    aVC.delegate = self;
    self.bvc = aVC;
    [self.navigationController pushViewController:aVC animated:YES];
}
BSFrameworks_Example[45051:2761346] == test == 
BSFrameworks_Example[45051:2761346] BSATransitionController dealloc
BSFrameworks_Example[45051:2761632] (null)==0x0 == 0x7fc8271096b0

相关文章

网友评论

    本文标题:iOS 关于 delegate 修饰词的错误认知

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