美文网首页iOS学习
@property 和 @synthesize 做了什么?

@property 和 @synthesize 做了什么?

作者: Cedric_Jc | 来源:发表于2017-10-16 15:52 被阅读26次

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

    重写了属性的 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 做了什么?

        本文链接:https://www.haomeiwen.com/subject/fwgkuxtx.html