以前在没有ARC(automatic reference counting)的时候我们使用assign与retain来修饰属性,后来引入了更安全的weak和strong来修饰属性
一、探讨assign、retain、strong、weak的区别
assign与weak
两者都是弱引用,assign通常用于普通类型属性(如int,NSInteger),常见的还会用于delegate对象的属性修饰,基本上来说两者是可以通用的。
Q:那对于delegate的属性修饰符到底是用assign好,还是weak好呢?
分析:weak在引用的对象被释放的时候会将delegate置为nil
,而assign修饰的delegate依然会指向原来的位置,这样在delegate引用的对象被释放后,delegate就会变成野指针。在OC中你给你一个nil对象发送消息不会crash,但是给一个对象(如:野指针)发送他不能响应的消息是会crash的,所以总的来说weak要比assign安全一些。
A:像delegate属性的修饰符建议用weak修饰而不是assign,这样你使用delegate的时候就不需要判断delegate对象是不是存在了。
retain和strong
他俩都是强引用,除了某些情况下不一样,比如修饰block,其他的时候也是可以通用的。
关于修饰block的时候,retain和strong有什么区别请看下面👇
二、探讨block的属性修饰符
提示一下,现在的Xcode创建的新项目都是默认采用ARC,我们如果想要对工程进行手动内存管理该怎么办?我们可以在TARGET -> Build Settings 下搜索Garbage Collection,会搜出Objective-C Automatic Reference Counting,将其设置为NO即可使整个工程处于MRC环境下了。那如果只想对某个类文件采用MRC,工程的其他类都用ARC呢,那我们可以在TARGET -> Build Phase -> Compile Sources下找到对应的类,双击修改Compiler flag,然后添加-fno-objc-arc
即可,反之在MRC工程下的某个类中添加-fobjc-arc
即可使用ARC。
MRC下
(1)用retain修饰block 且 不访问 external变量情况下
@interface ViewController ()
@property (nonatomic, retain) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
//num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSGlobalBlock__: 0x1008b9088>
(2)用retain修饰block 且 访问 external变量情况下
@interface ViewController ()
@property (nonatomic, retain) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSStackBlock__: 0x7ffeeb094988>
(3)用strong或copy修饰block 且 不访问 external变量情况下
@interface ViewController ()
@property (nonatomic, strong) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
//num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSGlobalBlock__: 0x10781c098>
(4)用strong或copy修饰block 且 访问 external变量情况下
@interface ViewController ()
@property (nonatomic, strong) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSMallocBlock__: 0x600002e4e5e0>
ARC下
(1)用retain修饰block 且 不访问 external变量情况下
@interface ViewController ()
@property (nonatomic, retain) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
//num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSGlobalBlock__: 0x10f4b30a0>
(2)用retain修饰block 且 访问 external变量情况下
@interface ViewController ()
@property (nonatomic, retain) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSMallocBlock__: 0x600002c78120>
(3)用strong或copy修饰block 且 不访问 external变量情况下
@interface ViewController ()
@property (nonatomic, strong) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
//num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSGlobalBlock__: 0x10ff380a8>
(4)用strong或copy修饰block 且 访问 external变量情况下
@interface ViewController ()
@property (nonatomic, strong) TestBlock block;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
__block NSInteger num = 1;
self.block = ^{
num = 2;
};
NSLog(@"---%@",self.block);
}
@end
打印结果:
---<__NSMallocBlock__: 0x60000075a550>
结果对照表(访问:指的是访问external变量,反之亦然):
retain 不访问 | strong(copy)不访问 | retain 访问 | strong(copy)访问 | |
---|---|---|---|---|
MRC | NSGlobalBlock | NSGlobalBlock | NSStackBlock |
NSMallocBlock |
ARC | NSGlobalBlock | NSGlobalBlock | NSMallocBlock | NSMallocBlock |
结论
- 只要不引入external变量,无论是
MRC
还是ARC
,无论是retain
还是strong
修饰block,block属性都是存在全局数据区 - 引入external变量时,retain和strong(copy)是有区别的,
在MRC
时,block存在栈区(stack)的,因此使用的时候要注意此时的block是否还存在,以免操作了野指针而闪退。
在ARC
时,block是存在堆区(heap)的。
总结
- 所以说block的属性修饰符应该用
strong
或copy
比较安全些。 - 对block来说,属性修饰符用
strong
或copy
效果是一样的。
【注意】上述提到的external变量
不是指C语言上的那个external变量,本处指的是block外面变量,可以是局部变量
,也是是一个属性
。