缘由是遇到了下面这个问题
重写了属性的
getter
和setter
方法后,编译器提示对应的实例变量不存在。
看了若干资料,理清了来龙去脉。
历史
在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
实例变量 提供 访问器方法。也可以理解为:指定 _foo
为foo
属性对应的实例变量
如果 _foo 实例变量没有被声明,@synthesize
会为类添加名为 _foo 的实例变量。
那 @synthesize foo;
做了什么事?
生成一个和 foo
同名的实例变量
@synthesize
实际上就相当于完成自动合成的过程,并且可以让程序员指定实例变量。
失效的自动合成
然而,Xcode 4.4 后,编译器进行的自动合成,是有可能会失效的。这种情况发生在当编译器判断程序要手动管理属性的时候,具体不会自动合成的场景如下
- 同时重写了get set
- 重写只读属性的 get
- 使用了@dynamic
- @protocol 中定义的属性
- category中定义的属性
- 重载的属性:当你在子类中重载父类的属性,你必须用
@synthesize
手动合成 ivar
在文章开头遇到的问题,就是因为我们重写了属性全部访问器方法后,让编译器使得自动合成失效,从而编译器不会自动编写访问器方法和生成Ivar
,于是我们在使用名为 下划线+属性 的实例变量的时候,就提示实例变量不存在。
解决方法
文章开头问题的解决方法:
使用 @synthesize foo = _foo
,合成 _foo
实例变量,并指定 foo 属性 为(四声) 它提供 get set 方法。
补充 @synthesize 合成实例变量的规则
- 如果指定了实例变量名称,就会生成一个指定了名称的成员变量。
@s foo = wtf_foo;//生成wtf_foo
- 如果这个实例变量存在,就不生成了。
@interface MyCls () {
NSString _foo;
}
@end
@implementation MyCls
@synthesize foo = _foo;
@end
- 如果没有指定实例变量名称,那就会自动生成一个和属性同名的实例变量。
@s foo; //就是生成 foo
补充 @dynamic
@dynamic 的意思就是跟系统说,不要创建 property 对应的成员变量(就是一般的 _someProperty),也不要自动生成 get/set 方法,同时不要报错,到在运行时我自己会来添加 get/set 方法。(转自【孙总的segmentfault】)