线程安全:atomic/nonatomic
在object-c中,生命属性时经常出现两个修饰符atomic和nonatomic,当不设置时默认时atomic。
系统会自动为这两个关键字修饰的属性初始化setter/getter方法。他们的区别在于
- atomic系统自动生成的getter/setter方法会进行加锁操作
- nonatomic系统自动生成的getter/setter方法不会进行加锁操作
总结就是atomic可以用于多线程,加锁保证了属性在多线程的情况下不出现数据错误。但是有了加锁这一层操作,就会消耗更多的系统性能和资源。nonatomic不适合用在多线程的情况下,相比atomic性能更好。
要注意的是
- atomic的锁仅仅保证了getter和setter存取方法的线程安全。
-
atomic不是线程安全的
atomic只是对属性的getter/setter方法进行了加锁操作,这种安全仅仅是set/get 的读写安全,并非真正意义上的线程安全,因为线程安全还有读写之外的其他操作(比如:如果当一个线程正在get或set时,又有另一个线程同时在进行release操作,可能会直接crash)
从代码实现上来看他们之间的区别
nonatomic
@synthesize value = _value;
//set
-(void)setValue:(int)value
{
if(_value != value)
{
_value = value;
}
}
//get
-(int)value
{
return _value;
}
atomic
@synthesize value = _value;
//set
-(void)setValue:(int)value
{
//同步代码块
@synchronized (self) {
if(_value != value)
{
_value = value;
}
}
}
//get
-(int)value
{
int val = 0; // 一般不用基础数据类型 默认为nil,在这里用int只是一个示范
//同步代码块
@synchronized (self) {
val = value
}
return val;
}
访问权限:readonly/readwrite
- readwrite: 编译器生成getter和setter,不声明默认为readwrite。
- readonly: 编译器只生成getter。通常的用法是,如果想将此属性暴露给其他类的话,在.h文件中,将其声明为readonly,在.m文件,将其声明为readwrite.这样的话,外部的其他文件就只能获取,不能修改。
内存管理(ARC):assign/weak/strong/copy
- strong: 默认为strong,每当使用该属性时增加引用计数。会引起循环引用当问题,遇到循环引用时用weak解决。
- assign: 常用于修饰int,long等原始类型。与weak类似, 简单赋值,不更改索引计数。
- weak: 使用时不会增加引用计数的指针。在其他类或者对象强引用此属性时,其会保持有一个有效的指针。如果没有其他对象要引用它的话,马上就会变为nil。这就避免了引用循环。
- copy: 与strong类似,不过它会复制对象的值,原对象不发生任何改变,在iOS开发中,一般copy关键字用在NSString、NSArray、NSDictionary等属性字段的修饰符。当原对象发生改变时不影响副本,有一定的安全性。
内存管理(MRC):assign/retain/copy/unsafe_unretained
- unsafe_unretained: 与weak类似,但是在没有其他对象进行引用时,并不会将值设置为nil
- retain: retain与strong为同样的功能,retain为ARC出现之前的关键字。
指定访问器方法:getter=/setter=
- getter=: 自定义访问器的getter名称。
- setter=: 自定义访问器的setter名称。
nonnull,nullable和null_resetable
为了对应swift的可选类型,object-c中增加了对应的修饰符。
- nullable: 提示可能为空
- nonnull: 不可为空
- null_resetable: setter可为空, gette不可为空