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
结果
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是一个道理,跟随控制器生命周期
- 场景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 的( 你也揪着它的小辫, 这样如果他爹松手了它也跑不了 )。