@property、@synthesize和@dynamic的区别

  • @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 相当做了两件事:

  1. 他原本的含义声明getter和setter方法;
  2. @synthesize nickname=_nickname;所以我们现在实例变量都是加“_”.这是一个很好的编程规范。

你可能感兴趣的:(@property、@synthesize和@dynamic的区别)