@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】)

你可能感兴趣的:(@property 和 @synthesize 做了什么?)