一、@property 本质
@property= ivar(实例变量) +getter/setter(存取方法);
在正规的 Objective-C 编码风格中,存取方法有着严格的命名规范。 正因为有了这种严格的命名规范,所以 Objective-C 这门语言才能根据名称自动创建出存取方法。
@property有两个对应的词,一个是 @synthesize,一个是 @dynamic。如果 @synthesize和 @dynamic都没写,那么默认的就是@syntheszie var = _var;
二、先来讲讲 “自动合成”(autosynthesis)
完成属性定义后,编译器会自动编写访问这些属性所需的方法,此过程叫做“自动合成”(autosynthesis)
需要强调的是,这个过程由编译器在编译期执行,所以编辑器里看不到这些“合成方法”(synthesized method)的源代码
例如
@interfacePerson:NSObject
@property NSString*firstName;
@property NSString*lastName;
@end
上述代码写出来的类与下面这种写法等效:
@interfacePerson:NSObject
- (NSString*)firstName;
- (void)setFirstName:(NSString*)firstName;
- (NSString*)lastName;
- (void)setLastName:(NSString*)lastName;
@end
除了生成方法代码 getter、setter 之外,编译器还要自动向类中添加适当类型的实例变量,并且在属性名前面加下划线,以此作为实例变量的名字。
在前例中,会生成两个实例变量,其名称分别为 _firstName 与 _lastName。
也可以在类的实现代码里通过 @synthesize 语法来指定实例变量的名字.
@implementationPerson
@synthesizefirstName = _myFirstName;
@synthesizelastName = _myLastName;
@end
我们每次在增加一个属性,系统都会在 ivar_list 中添加一个成员变量的描述,在 method_list 中增加 setter 与 getter 方法的描述,在属性列表中增加一个属性的描述,然后计算该属性在对象中的偏移量,然后给出 setter 与 getter 方法对应的实现,在 setter 方法中从偏移量的位置开始赋值,在 getter 方法中从偏移量开始取值,为了能够读取正确字节数,系统对象偏移量的指针类型进行了类型强转.
三、@synthesize的作用
是如果你没有手动实现 setter 方法和 getter 方法,那么编译器会自动为你加上这两个方法
如果 @synthesize和 @dynamic都没写,那么默认的就是@syntheszie var = _var;
四、什么情况下不会autosynthesis(自动合成)
同时重写了 setter 和 getter 时
重写了只读属性的 getter 时
使用了 @dynamic 时
在 @protocol 中定义的所有属性
在 category 中定义的所有属性
重载的属性,当你在子类中重载了父类中的属性,你必须使用 @synthesize 来手动合成ivar
五、@dynamic的作用
@dynamic 告诉编译器:属性的 setter 与 getter 方法由用户自己实现,不自动生成。
当然对于 readonly 的属性只需提供 getter 即可
假如一个属性被声明为 @dynamic var,然后你没有提供 @setter方法和 @getter 方法,编译的时候没问题,但是当程序运行到 instance.var = someVar,由于缺 setter 方法会导致程序崩溃
或者当运行到 someVar = var 时,由于缺 getter 方法同样会导致崩溃。编译时没问题,运行时才执行相应的方法,这就是所谓的动态绑定
例如:
#import
@interface People : NSObject
@property(nonatomic, copy)NSString*name;
@end
#import "People.h"
@implementation People
@dynamic name;
@end
当执行到:
People *p= [People new];
p.name = @"Peter";
程序就会crash,原因是“[People setName:]: unrecognized selector sent to instance”。
解决这种奔溃的方法有三种:
方法一:
最简单粗暴,但是还是挺管用,直接注释掉@dynamic name这行代码即可,由编译器自动添加。
但是如果我们不想让编译器自动添加,那么我们可以手动添加或则在运行时添加都可以。
方法二:
手动添加,由于@dynamic不能像@synthesize那样向实现文件(.m)提供实例变量,所以我们需要在类中显式提供实例变量。
#import
@interface People : NSObject
@property (nonatomic, copy) NSString *name;
@end
#import "People.h"
@interface People ()
{
NSString *_name;
}
@implementation People
@dynamic name;
- (void)setName:(NSString *)name {
_name = [name copy];
}
- (NSString *)name {
return _name;
}
@end
方法三:
通过runtime机制在运行时添加属性的存取方法。
在C函数中不能直接使用实例变量,需要将Objc对象self转成C中的结构体,因此在Person类同样需要显式声明实例变量而且访问级别是@public
#import
@interface People : NSObject
@property (nonatomic, copy) NSString *name;
@end
#import "People.h"
#import
#import
@interface People ()
{
@public
NSString*_name;
}
@end
@implementation People
@dynamic name;
+ (BOOL)resolveInstanceMethod:(SEL)sel {
if(sel ==@selector(setName:)) {
class_addMethod([selfclass], sel, (IMP)setName,"v@:@");
returnYES;
}elseif(sel ==@selector(name)){
class_addMethod([selfclass], sel, (IMP)getName,"@@:");
returnYES;
}
return [super resolveInstanceMethod:sel];
}
void setName(idself,SEL_cmd,NSString* name)
{
if(((People*)self)->_name!= name) {
((People*)self)->_name= [namecopy];
}
}
NSString* getName(idself,SEL_cmd)
{
return((People*)self)->_name;
}
@end
参考原文链接:https://www.jianshu.com/p/c883687c6405