iOS 属性

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • iOS开发中属性修饰符基本上每天都会打交道,网上总结也很多,本文按照实际开发的角度简单介绍一下属性修饰符,属性修饰...
    FlyElephant阅读 2,308评论 0 4
  • 人的一生会有多少次遇见才会成为相识?于千万人中与你相遇相知是一件很不易的事情。所以,珍惜眼前人,这是缘分的遇见。
    喃呢龟阅读 296评论 0 0
  • 共享是有趣的事,经济是有利的事(利润的利),所以在我看来,共享经济是一种有趣的商业模式。将你闲置的资源共享给别人,...
    好学长松哥阅读 385评论 0 0
  • 泡泡 我家养了一只猫,它的毛是浅黄的,四只小脚...
    Qing疯阅读 141评论 0 0
  • 六月是怒放的盛夏,毕业是永不褪色的光华!点点滴滴的过往,深深扎根在心田,化作浓浓的爱意、依依的不舍,泛滥成了泪,滴...
    午后窗台的猫阅读 301评论 0 0