理解"属性"这一概念

"属性"是OC的一项特性,用于封装对象的数据.OC对象通常会把其所需要的数据保存为各种实例变量.你也许知道"属性"概念,但是你未必知道其全部细节.而且还有许多和属相相关的麻烦事儿.

比如,某个代码库中的代码使用了一份旧的类定义,如果和其相链接的代码使用了新的类定义.那么运行时就会出现不兼容的现象.各种编程语言都有应对此问题的办法.OC的做法是,把实例变量当做一种存储偏移量所用的"特殊变量",交给"类对象"保管.偏移量会在运行期查找,如果类的定义变了,那么存储的偏移量就变了,这样的话,无论何时访问实例变量,总能使用正确的偏移量.甚至可以在运行期向类中新增实例变量.这就是稳固的"应用程序二进制接口"(ABI).ABI定义了许多内容,其中一项就是生成代码时所应遵循的规范.

要访问属性,可以使用"点语法",在纯C中,如果想访问分配在栈上的struct结构体里面的成员,也需要使用类似的语法,编译器会把"点语法"转换成对存取方法的调用.使用"点语法"的效果与直接调用存取方法相同.

若不想令编译器自动合成存取方法,则可以自己实现,如果你只实现了其中一个存取方法,那么另外一个还是会由编译器来合成的.还有一种办法能阻止编译器自动合成存取方法,就是使用@dynamic关键字,它会告诉编译器:不要自动创建实现属性所用的实例变量.也不要为其创建存取方法.而且,在编译访问属性的代码时,及时编译器发现,没有定义存取方法.也不会报错.它相信这些办法能在运行期找到.

@interface CWGPerson : NSManagedObject
@property NSString *firstName;
@property NSString *lastName;
@end

@implementation CWGPerson
@dynamic firstName, lastName;
@end

编译器不会为上面这个类自动合成存取方法或实例变量.如果用代码访问其中的属性,编译器也不会发出警示信息.

内存管理语义

属性用于封装数据,而数据则要有"具体的所有权语义".下面这一组特性仅会影响"设置方法".例如:用"设置方法"设定一个新值时,它是应该"保留"此值呢,还是只将其赋给底层实例变量就好了?编译器在合成存取方法时,要根据此特质来决定所生成的代码.如果自己编写存取方法,那么就必须同有关属性所具备的特质相符.

  • assign "设置方法"只会执行针对"纯量类型"的简单赋值操作.
  • strong 此特质表明该属性定义了一种"拥有关系".为这种属性设置新值时,设置方法会先保留新值,并释放旧值,然后再讲新值设置上去.
  • weak 此特质表明该属性定义了一种"非拥有关系".为这种属性设置新值时,设置方法既不会保留新值,也不会释放旧值,此特质同assign类似,然而在属性所指的对象遭到摧毁时,属性值也会清空.
  • unsafe_unretained 此特质的语义和assign相同,但是它适用于"对象类型".该特性表达一种"非拥有关系",当目标对象遭到摧毁时,属性值不会自动清空,这是和weak的区别.
  • copy 此特质所表达的所属关系和strong类似,然而设置方法并不保留新值,而是将其拷贝,当属性类型为NSString*,经常用此特质来保护其封装性,因为传递给设置方法的新值有可能指向一个NSMutableString类的实例.这个类是NSString的子类,表示一种可以修改其值的字符串,此时若是不拷贝字符串,那么设置完属性之后,字符串的值就可能会在对象不知情的情况下遭人修改.所以,这时就要拷贝一份"不可变"的字符串,确保对象中的字符串值不会无意间变动.只要实现属性所用的对象是"可变的",那就应该在设置新属性时拷贝一份.
方法名
  • getter=<name> 指定"获取方法"的方法名.如果某个属性是Boolean型,而你想为其获取方法加上"is"这种一看就明白的前缀,那就可以用这个办法指定.
@property (nonatomic, getter = isOn) Bool on;
  • setter=<name> 指定"设置方法"的方法名.这种做法几乎用不到.

通过上诉特质,可以微调由编译器所合成的存取方法.不过需要注意:若是自己来实现这些存取方法,那么应该保证其具备相关属性所声明的特质.这一点很重要.比方说:某个属性声明为copy,那么就应该在"设置方法"中拷贝相关对象,否则会误导该属性的使用者.而且,若是不遵守这个约定,还会令程序产生bug.如果想在其他方法里设置属性值,那么同样要遵守属性定义中宣称的语义.

@interface CWGPerson : NSManagedObject
@property NSString *firstName;
@property NSString *lastName;

- (void)initWithFirstName:(NSString *)firstName
                        lastName:(NSString *)lastName;
@end

@implementation CWGPerson
- (void)initWithFirstName:(NSString *)firstName
                        lastName:(NSString *)lastName
{
    if(self = [super init]) {
     _firstName = [firstName copy];
     _lastName = [lastName copy];
  }
   return self;
}
@end

也许你会问:为什么不调用属性所对应的"设置方法"呢?如果用了"设置方法",不是总能保证准确的语义吗? 我将会在后面的文章中详细介绍为什么绝不应该在init(或dealloc)方法中调用存取方法.

总结:

  • 可以用@property语法定义对象中所封装的数据.
  • 通过"特质"来指定存储数据所需的正确语义
  • 在设置属性所对应的实例变量时,一定要遵从该属性所声明的语义.
  • 在iOS开发中,应该使用nonatomic, 因为atomic属性会严重影响性能.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容