iOS控件中的weak与strong

0x0.问题导入:weak与strong在控件定义时的纠结

weak与strong在控件定义过程中时常用到,通常使用xib拖动出来的使用weak,纯代码编写也有使用weak也有strong,如下:

@property (weak, nonatomic) IBOutlet UILabel *weakLabel;
@property (weak, nonatomic) UILabel *weakLabel2;
@property (strong, nonatomic) UILabel *strongLabel;

这三种各自的区别在哪,接着来实验一下:

@interface ViewController ()
@property (weak, nonatomic) IBOutlet UILabel *weakLabel;
@property (weak, nonatomic) UILabel *weakLabel2;
@property (strong, nonatomic) UILabel *strongLabel;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    //控件初始化
    UILabel *labTemp = [[UILabel alloc]init];
    labTemp.frame = CGRectMake(50, 60, 100, 21);
    labTemp.text = @"weakLabel2";
    self.weakLabel2 = labTemp;
    //如果labTemp没有addSubview到self.view中,_weakLabel2将在viewDidLoad执行完后释放
    [self.view addSubview:labTemp];
    
    [self.view addSubview:self.strongLabel];
    
    /*
     代码实现控件弱引用,提示: ("Warning: Assigning retained object to weak variable; object will be released after assignment")
     这是因为对象没有拥有者,在创建之后就会被立即释放
    */
//    _weakLabel2 = [[UILabel alloc]init];
//    _weakLabel2.frame = CGRectMake(50, 60, 100, 21);
//    _weakLabel2.text = @"weakLabel2";
//    [self.view addSubview:_weakLabel2];
    
    //1.对三种定义方式进行移除
    [self.weakLabel removeFromSuperview];
    [self.weakLabel2 removeFromSuperview];
    [self.strongLabel removeFromSuperview];
}

-(void)viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];
    //2.移除后进行添加[之所以放这方法测试,是因为remove后有释放延迟]
    [self.view addSubview:_weakLabel];
    [self.view addSubview:_weakLabel2];
    [self.view addSubview:_strongLabel];

    NSLog(@"_weakLabel --> %@",_weakLabel);
    NSLog(@"_weakLabel2 --> %@",_weakLabel2);
    NSLog(@"_strongLabel --> %@",_strongLabel);
}

#pragma mark - getter
-(UILabel*)strongLabel{
    if(!_strongLabel){
        _strongLabel = [[UILabel alloc]init];
        _strongLabel.text = @"strongLabel";
        _strongLabel.frame = CGRectMake(50, 100, 100, 21);
    }
    return _strongLabel;
}
@end

结果


t1.png

0x1.weak与strong关键字特性

弱引用weak,即当对象在释放之后,指针地址被置为nil
强引用strong,进行retain+1操作,对象持有

0x2.controller中控件生命周期

  • IB中outlet的控件:UIViewController->UIView->subView->UILabel
    @property(null_resettable, nonatomic,strong) UIView *view;
    UIViewController中self.view是个strong类型,当系统将
    @property (weak, nonatomic) IBOutlet UILabel *weakLabel;
    add到self.view时,也就是说weakLabel已经被强引用一次,所以一般不再使用strong进行修饰;当控制器释放时,view也会被释放,weakLabel同样。

  • weak类型的控件:UIViewController->UIView->subView->UILabel
    @property (weak, nonatomic) UILabel *weakLabel2;
    其实就是IB的代码编写的实现,内存管理方式是一样的

  • strong类型的控件:
    UIViewController->UIView->subView->UILabel
    UIViewController->UILabel
    @property (strong, nonatomic) UILabel *strongLabel;
    因此当strongLabel在view移除后,controller还对其进行保持

0x3.weak与strong场景使用

- 场景1:IB中多状态UI布局方式(所以IB还是个好东西,可视化程度高,节省很多时间)

//@property (strong, nonatomic) IBOutlet UIView *statusView1;
//@property (strong, nonatomic) IBOutlet UIView *statusView2;
@property (weak, nonatomic) IBOutlet UIView *statusView1;
@property (weak, nonatomic) IBOutlet UIView *statusView2;

这里就会出现两种情况:
一是主容器view里面的label可以直接为weak,其它两个view也为weak
当状况view1、2初始化完成后,在viewDidLoad方法里面没有add到self.view中,当数据load回来要更新状态时,会发现view1、2已经为nil了
二是其它两个view为strong
这种情况就很明了,跟代码strong是一个道理,跟随控制器生命周期

iOS控件中的weak与strong_第1张图片
t4.png

- 场景2:weak与strong代码布局使用
1.懒加载,使用strong。
为什么不使用weak,getter实现的时候,对象不持有。

@property (strong, nonatomic) UILabel *strongLabel;
  -(UILabel*)strongLabel{
    if(!_strongLabel){
        _strongLabel = [[UILabel alloc]init];
        _strongLabel.text = @"strongLabel";
        _strongLabel.frame = CGRectMake(50, 100, 100, 21);
    }
    return _strongLabel;
}

2.weak定义
@property (weak, nonatomic) UILabel *weakLabel2;
注: 如果labTemp没有addSubview到self.view中,_weakLabel2将在viewDidLoad执行完后释放

 UILabel *labTemp = [[UILabel alloc]init];
 labTemp.frame = CGRectMake(50, 60, 100, 21);
 labTemp.text = @"weakLabel2";
 self.weakLabel2 = labTemp;
 [self.view addSubview:labTemp];

最后

引用知乎的一句话 余畅
因为控件他爹( view.superview )已经揪着它的小辫了( strong reference ),你( viewController )眼瞅着( weak reference )就好了。
当然,如果你想在 view 从 superview 里面 remove 掉之后还继续持有的话,还是要用 strong 的( 你也揪着它的小辫, 这样如果他爹松手了它也跑不了 )。

你可能感兴趣的:(iOS控件中的weak与strong)