@property 和 @synthesize 做了什么?

缘由是遇到了下面这个问题

重写了属性的 gettersetter 方法后,编译器提示对应的实例变量不存在。

看了若干资料,理清了来龙去脉。

历史

在Xcode 4.4 之前,我们在 .h 中写完@proprety foo;之后,都会在 .m 中写上 @synthesize foo = _foo;来为属性生成访问方法。

然后在Xcode 4.4 之后,我们只要写了 @proprety foo; 就可以自动生成属性的访问方法。

具体原因,我们从@proprety说起。

@proprety

我们知道,写下@proprety 后,编译器实际上是做了两件事,生成getter 和 setter,给类中添加名为属性名前加下划线的实例变量。这个过程叫做自动合成

@synthesize

然后@synthesize foo = _foo做了什么事情呢?

foo 属性帮助 _foo 实例变量 提供 访问器方法。也可以理解为:指定 _foofoo 属性对应的实例变量

如果 _foo 实例变量没有被声明,@synthesize 会为类添加名为 _foo 的实例变量。

@synthesize foo;做了什么事?

生成一个和 foo 同名的实例变量

@synthesize实际上就相当于完成自动合成的过程,并且可以让程序员指定实例变量。

失效的自动合成

然而,Xcode 4.4 后,编译器进行的自动合成,是有可能会失效的。这种情况发生在当编译器判断程序要手动管理属性的时候,具体不会自动合成的场景如下

  1. 同时重写了get set
  2. 重写只读属性的 get
  3. 使用了@dynamic
  4. @protocol 中定义的属性
  5. category中定义的属性
  6. 重载的属性:当你在子类中重载父类的属性,你必须用 @synthesize 手动合成 ivar

在文章开头遇到的问题,就是因为我们重写了属性全部访问器方法后,让编译器使得自动合成失效,从而编译器不会自动编写访问器方法和生成Ivar,于是我们在使用名为 下划线+属性 的实例变量的时候,就提示实例变量不存在

解决方法

文章开头问题的解决方法:

使用 @synthesize foo = _foo ,合成 _foo 实例变量,并指定 foo 属性 为(四声) 它提供 get set 方法。

补充 @synthesize 合成实例变量的规则

  1. 如果指定了实例变量名称,就会生成一个指定了名称的成员变量。@s foo = wtf_foo;//生成wtf_foo
  2. 如果这个实例变量存在,就不生成了。
@interface MyCls () {
      NSString _foo;
}
@end
@implementation MyCls
@synthesize foo = _foo;  
@end
  1. 如果没有指定实例变量名称,那就会自动生成一个和属性同名的实例变量。@s foo; //就是生成 foo

补充 @dynamic

@dynamic 的意思就是跟系统说,不要创建 property 对应的成员变量(就是一般的 _someProperty),也不要自动生成 get/set 方法,同时不要报错,到在运行时我自己会来添加 get/set 方法。(转自【孙总的segmentfault】)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,659评论 19 139
  • 刚从校园里出来的时候,很不适应社会生活,一心觉得自己堂堂大学生,不该这么被社会摧残。不说要被捧在掌心,至少要公平对...
    兰花的兰阅读 404评论 9 7
  • 《易经》里面说:“天行健,君子以自强不息;地势坤,君子以厚德载物。”于是就由康有为先生的演说衍生出了清华大...
    奔跑的风筝阅读 403评论 0 0

友情链接更多精彩内容