1.@property 属性
在我们声明property这个关键字的时候例如:
@property (nonatomic, strong) NSMutableArray *dataArray;
这个时候默认编译器会帮我们声明一个Ivar(带下划线的成员变量_dataArray),还有对应的setter和getter方法,使用了property之后,就可以使用iOS中的点语法,可以通过self.dataArray来使用这个实例变量,实质上也是调用了对应的setter和getter方法。
@property的特性,@property还有一些关键字,它们都是有特殊作用的,比如上述代码中的nonatomic,strong.我把它们分为三类,分别是:原子性,存取器控制,内存管理。
原子性:
atomic(默认):atomic意为操作是原子的,意味着只有一个线程访问实例变量。atomic是线程安全的,至少在当前的存取器上是安全的。但是并不意味着它是线程安全的,它会增加正确的几率,能够更好的避免线程的错误,但是它仍然是线程不安全的。它是一个默认的特性,但是很少使用,因为比较影响效率,这跟ARM平台和内部锁机制有关。
nonatomic:nonatomic跟atomic刚好相反。表示非原子的,可以被多个线程访问。它的效率比atomic快。但不能保证在多线程环境下的安全性,在单线程和明确只有一个线程访问的情况下广泛使用。
存取器控制
readwrite(默认):readwrite是默认值,表示该属性同时拥有setter和getter。
readonly: readonly表示只有getter没有setter。
有时候为了语意更明确可能需要自定义访问器的名字:
@property (nonatomic, setter = mySetter:,getter = myGetter ) NSString *name;
最常见的是BOOL类型,比如标识View是否隐藏的属性hidden。可以这样声明:
@property (nonatomic,getter = isHidden ) BOOL hidden;
内存管理
strong 释放旧对象将旧对象的值赋予输入对象,再提高输入对象的索引计数为1,此关键字经常使用。
weak 不增加引用计数,不持有对象,因此也不能决定对象释放 对比assign 的一个好处是,当对象消失时指针自动归为nil
assign 适用于基础数据类型(NSInteger CGFloat...)不增加引用计数
copy 建立一个索引计数为1 的对象然后释放旧对象 此属性只对那些实行了NSCopying协议的对象类型有效(NSString , Block)
如果声明属性的时候不显式指定任何关键字的时候默认的关键字分两种,一种是基本数据类型的默认关键字是atomic,readwrite,assign,第二种对于普通的OC对象,默认关键字是atomic,readwrite,strong。
同时重写setter/getter的问题:我们会发现,当我们同时重写setter/getter时会报错,为什么呢?这是因为当我们同时重写setter/getter时,编译器自动添加的代码@synthesize dataArray = _dataArray;失效,就不会自动为我们生成实例变量_dataArray了,setter/getter操作的对象就不存在了。所以我们要加上@synthesize dataArray = _dataArray;,手动指定setter/getter要操作的实例对象是_dataArray.
2.@synthesizer
实现property所声明的方法的定义。其实说直白就像是:property声明了一些成员变量的访问方法,synthesize则定义了由property声明的方法。
他们之前的对应关系是:property 声明方法 ->头文件中申明getter和setter方法 synthesize定义方法 -> m文件中实现getter和setter方法。
@interface ViewController ()
@property (nonatomic, strong) UIButton *myButton;
@end
@implementation ViewController
@synthesize myButton ;
@synthesize 语句只能被用在 @implementation 代码段中,@synthesize 还有一个作用,可以指定与属性对应的实例变量,例如@synthesize myButton = xxx;那么self.myButton其实是操作的实例变量xxx,而不是_myButton了。如果默认没写@synthesize和 @dynamic都没写,那么默认的就是@syntheszie var = _var;
3.@dynamic
@dynamic 告诉编译器:属性的 setter 与 getter 方法由用户自己实现,不自动生成。(当然对于 readonly 的属性只需提供 getter 即可)。假如一个属性被声明为 @dynamic var,然后你没有提供 @setter方法和 @getter 方法,编译的时候没问题,但是当程序运行到 instance.var = someVar,由于缺 setter 方法会导致程序崩溃;或者当运行到 someVar = var 时,由于缺 getter 方法同样会导致崩溃。编译时没问题,运行时才执行相应的方法,这就是所谓的动态绑定。
参考:
http://blog.eddie.com.tw/2010/12/08/property-and-synthesize/
http://www.cocoachina.com/bbs/read.php?tid=7322
http://www.cnblogs.com/pinping/archive/2011/08/03/2126150.html