这篇文章分三段,先讲@property 以及 @synthesize ,然后到self.property 与 _property 的区别。
参考来自:《Objective-C编程》17.1存取方法 P103
《Objective-C 程序设计》第六版 10.2.1 关于属性,存取方法和实例变量 P203
《Effective Objective-C 2.0》理解属性这一概念 P21
@property
- 当类中的实例变量需要在外部被使用时,我们需要为这个实例变量编写存取方法(因为面向对象的封装性,外部不能直接访问实例变量)。
- 需要被外部用到的实例变量可能有很多,于是Apple创建了一条便捷途径,称为属性(property)。通过属性,可以用一行代码来同时声明存取方法。
举例:
- @property NSString *yhx;
- -(NSString *) yhx ;//供外部读取实例变量的方法
-(void) setYhx: (NSString *) yhx ;//供外部设置实例变量的方法
两个作用是一样的,都是在接口文件中声明了存取方法。( 是这样吗?)
@synthesize
当在接口文件(.h文件)中声明了@property后,在实现文件(.m文件)中,会自动合成存取方法,等效于自动帮你写了以下这句,但是不会显示给你看:
@synthesize yhx = _yhx ;//并不会显示出来
在这里,我现在的理解是这样的
- “yhx” 指的是存取方法的名字
- “_yhx” 是实例变量的名字(给实例变量命名时,习惯在最前面加一个下划线)
这里做的是
- 先声明一个名为 “_yhx” 的实例变量( 是这样吗? 嗯,应该是如此,见本文最后)
- 再把名字为 “yhx” 的存取方法与名为 “_yhx” 的实例变量联系了起来
- “yhx”方法是用合适的步骤来读取 “ _yhx” 这个实例变量的值。
self.property 与 _property 的区别
那么
self.yhx ; //通过@property自动生成的存取方法,或者自定义的存取方法来获得yhx的值
_yhx ; //直接访问实例变量的值
两者的区别在于,通过存取方法访问比直接访问多做了一些其他的事情(例如内存管理,复制值等)。
在斯坦福iOS7课程的课件中,是这样说的:
直接访问,应只用在存取方法以及初始化方法当中。
原文见补充内容。
我的理解是,因为在写存取方法的时候,可能要自定义一些附加的任务,使得存取能够符合自己的要求,比如
- @property (copy) NSString *yhx; 这里声明了copy特性,所以存的时候就不是直接赋值了,而是先copy,再赋值。 (详情可见《Effective Objective-C 2.0》中的第6条)
** P.S ** 当开发者自己把存和取,两个方法都自己写好了,那么
- 不会自动合成存取方法了
- 不能用“_yhx”直接访问实例变量
- 需要自己把 @synthesize yhx = _yhx; 写出来,才能用 “_yhx” 直接访问实例变量
如果自定义了存或取方法,那么会自动合成另一个方法,也能用“_yhx”直接访问实例变量。
补充内容
编译器怎么处理@synthesize
以下内容参考于 iOS @synthesize var = _var 变量前置下划线解释
@synthesize age=_age;
下面是将被编译的代码:
@interface User : NSObject {
NSInteger _age;
}
@end
@implementation User
-(void)setAge:(NSInteger)newAge {
_age=newAge;
}
-(void)age {
return _age;
}
@end