属性关键字:
ARC
- strong
- weak
MRC
- retain
不分环境
- copy
- assign
- nonatomic
- atomic
- readonly
- readwrite
- getter
- setter
- unsafe_unretained
- class
对比说明
assign 、weak和unsafe_unretained:
三者都不会对对象的引用计数+1,assign一般用来修饰基本数据类型,例如double int等,unsafe_unretained与assign类似,weak不能修饰基本数据类型,只能修饰对象类型。
weak相对于assgin,unsafe_unretained在使用上更安全不会出现野指针情况,因为weak在指向的对象释放后会被自动赋值为nil。
strong 和 retain
strong和retain可以认为是等价的,都会让对象引用计数+1,区别是strong是在arc下,retain是在mrc下。
//官方代码示例
// The following declaration is a synonym for: @property(retain) MyClass *myObject;
@property(strong) MyClass *myObject;
copy
不能修饰基本数据类型属性,用来修饰对象。不会对引用计数造成+1,会复制一个相同的对象。例如常用来修饰NSString 类型。注释:自定义对象需要实现NSCoping协议
nonatomic和atomic
原子性和非原子性,atomic原子性在生成set get 方法时会考虑线程问题,增加多线程保护,有加锁操作,但是会使属性读写性能降低。nonatomic非原子性不考虑安全问题,没有加锁操作,访问速度快,一般我们使用nonatomic,系统默认是atomic。atomic是线程安全,其实这个说法是不准确的,atomic只是对属性的getter/setter方法进行了加锁操作,这种安全仅仅是set/get 的读写安全,并非真正意义上的线程安全,因为线程安全还有读写之外的其他操作(比如:如果当一个线程正在get或set时,又有另一个线程同时在进行release操作,可能会直接crash)
@property(atomic,copy) NSString *str2;
@property(nonatomic,copy) NSString *str3;
-(void)setStr2:(NSString *)str2{
@synchronized (self) {
if (_str2 != str2) {
_str2 = nil;
_str2 = [str2 copy];
}
}
}
-(NSString*)str2{
NSString *str = nil;
@synchronized (self) {
str = _str2;
}
return str;
}
-(void)setStr3:(NSString *)str3{
if (_str3 != str3) {
_str3 = nil;
_str3 = [str3 copy];
}
}
-(NSString*)str3{
return _str3;
}
readonly和readwrite
通过字面意思也可以理解readonly属性是只读属性,即系统只生成get方法。readwrite是读写属性,即系统自动生成set get,readwrite为系统默认参数。
getter和setter
可以重命名生成set get 方法名字。
@property(nonatomic,copy,setter=testString4:) NSString *str4;
@property(nonatomic,copy,getter=ggString5) NSString *str5;
- (void)testString4:(NSString *)str4{
_str4 = [str4 copy];
}
- (NSString *)ggString5{
return _str5;
}
class
class 使用不常见,可以使属性通过类名加点语法获取。
static NSString *classStr = nil;
@property(nonatomic,class) NSString *string;
+(void)setString:(NSString *)string{
classStr = string;
}
+ (NSString *)string{
return classStr;
}
NSLog(@"%@",OBJProperty.string);