iOS中assign与weak,retain与strong的区别以及block的属性修饰符探讨

以前在没有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的属性修饰符应该用strongcopy比较安全些。
  • 对block来说,属性修饰符用strongcopy效果是一样的。

【注意】上述提到的external变量不是指C语言上的那个external变量,本处指的是block外面变量,可以是局部变量,也是是一个属性

你可能感兴趣的:(iOS中assign与weak,retain与strong的区别以及block的属性修饰符探讨)