@property:只是对getter和setter方法进行了声明,其他什么也没干。
@synthesize:编译期间,让编译器自动生成getter和setter方法,如果已经手动生成了getter或setter方法,系统会屏蔽该方法的实现。
如果没有实例变量的话自动添加实例变量,实现主要是根据@property
特性的设置,例如property设置特性为 copy(retain)、readonly、assgin等。@dynamic:告诉编译器,不自动生成getter和setter方法,由开发者手动生成,或存取方法在运行时动态创建绑定(主要使用在CoreData实现NSManagedObject子类时使用,由Core Data框架在程序运行时动态生成子类属性),避免编译期间产生警告。
对于只读属性需要提供 setter方法,对于读写属性需要提供 setter 和 getter方法。
如果没有自己生成,当在调用getter或者setter方法的时候,程序就会发生crash。
头文件中:
@interface Person : NSObject
@property(nonatomic, copy) NSString *name;
@property(nonatomic, assign) NSInteger age;
@end
1. 实现文件中:
@implementation Person
@synthesize name;
@synthesize age;
实现文件中这样写,确实很完美,这样我们就很自然的生成 name、age 实例变量的getter 和setter方法。
2. 改变实文件中的写法
@implementation Person
@synthesize name = _name;
@synthesize age;
在这里需要注意了,@synthesize name = _name;
这句话我们分为三部分来看:
- 第一部分:
@synthesize:
oc的关键字。 - 第二部分:
name:
这里的name
是指的@property
的后面那个变量的标识,如果我把头文件中的@property(nonatomic, copy)NSString *name;
这句话中的name
改变为nickName
,那么@synthesize
中的name
也应该相应的改变为nickName
,否则就会出错。 - 第三部分:
_name
,可能你会在这里迷惑,为什么会出来一个_name
的标识呢?而且还没有出错。首先我们要明确的是这第三部分是是指实例变量。也就说那个实例变量要与@property
相对应。可能你会发现我的头文件中并没有_name
的实例变量啊。这样难道不会报错吗?答案是不会的因为@synthsize
有自动创建实例变量的能力。如果编译器在实例变量列表中没有看找见_name
那么编译器就会自动的创建一个_name
实例变量。这样我们在程序别的地方可以调用实力变量:_name = @"hello";
这样是正确的。
3. 更改实现文件
@implementation Person
@dynamic name;
如果这样写。就是说name的getter方法和setter方法没有实现。如果你在程序中调用 self setName:
这个方法会报错。
总结:
其实@Property
只是说明了 我们声明一个方法。而@synthesize
告诉我们 方法中的主角是谁,也就说要给那个实例变量赋值。
注意:
Xcode 6之后我们写程序可能只需要写@property (nonatomic, retain)NSString *nickName;
并没有发现什么@synthesize
,那是因为现在一句@property
相当做了两件事:
- 他原本的含义声明getter和setter方法;
- @synthesize nickname=_nickname;所以我们现在实例变量都是加“_”.这是一个很好的编程规范。