1.@property有兩個(gè)對(duì)應(yīng)的詞,一個(gè)是 @synthesize,一個(gè)是 @dynamic。如果 @synthesize和 @dynamic都沒(méi)寫(xiě),那么默認(rèn)的就是@syntheszie var = _var搓译,也就是自動(dòng)給我們生成成員變量,但如果我們同時(shí)手動(dòng)實(shí)現(xiàn)了setter和getter方法锋喜,property就不會(huì)自動(dòng)生成成員變量侥衬,其他的情況下會(huì)自動(dòng)生成成員變量
2.@synthesize的語(yǔ)義是如果你沒(méi)有手動(dòng)實(shí)現(xiàn)setter和getter方法,那么編譯器會(huì)自動(dòng)為你加上這兩個(gè)方法跑芳。在Xcode4.5和以后的版本中轴总,可以省略@synthesize,編譯器會(huì)自動(dòng)加上setter和getter方法的實(shí)現(xiàn)。并且默認(rèn)會(huì)去訪問(wèn)_age這個(gè)成員變量博个,如果找不到_age這個(gè)成員變量怀樟,會(huì)自動(dòng)生成一個(gè)叫做_age的私有成員變量。如果在.m文件中同時(shí)實(shí)現(xiàn)getter和setter的時(shí)候需要在.m文件中定義@synthesize age = _age
3.@dynamic告訴編譯器盆佣,屬性的setter和getter方法由用戶自己實(shí)現(xiàn)往堡,不自動(dòng)生成,(當(dāng)然對(duì)于readonly的屬性只需要提供getter方法即可)共耍。假如一個(gè)屬性被聲明為@dynamic var ,然后你沒(méi)有提供@setter和getter方法虑灰,編譯的時(shí)候沒(méi)問(wèn)題,但是當(dāng)程序運(yùn)行到instanve.var = someVar,由于我們沒(méi)有實(shí)現(xiàn)setter方法痹兜,程序會(huì)崩潰穆咐,或者當(dāng)運(yùn)行到someVar = var時(shí),由于缺getter方法同樣會(huì)導(dǎo)致程序崩潰字旭,編譯時(shí)沒(méi)問(wèn)題对湃,運(yùn)行時(shí)才執(zhí)行相應(yīng)的方法,這就是所謂的動(dòng)態(tài)綁定遗淳。那么如何實(shí)現(xiàn)呢拍柒?這里有兩種方法,第一種是由我們自己實(shí)現(xiàn)存取方法屈暗,第二種則是存取方法在運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建并綁定拆讯,我們來(lái)看一下代碼如何實(shí)現(xiàn)
通過(guò)私有變量來(lái)實(shí)現(xiàn)@dynamic的存取方法
// .h
@interface BRTeacher : NSObject{
@private
__strong NSString *_teacherName;
}
@property (nonatomic,copy) NSString *teacherName;
@end
//.m
@dynamic teacherName;
- (void)setTeacherName:(NSString *)newValue{
_teacherName = newValue;
}
- (NSString *)teacherName{
return _teacherName;
}
通過(guò)消息轉(zhuǎn)發(fā)來(lái)實(shí)現(xiàn)@dynamic的存取方法
//.h
@interface BRTeacher : NSObject{
@private
__strong NSString *_teacherName;
NSMutableDictionary *_propertiesDict;
}
@property (nonatomic,copy) NSString *teacherName;
@property (nonatomic,copy) NSString *schoolName;//學(xué)校名稱
//.m
@dynamic teacherName;
@dynamic schoolName;
- (id)init{
self = [super init];
if(self){
_propertiesDict = [[NSMutableDictionary alloc] init];
}
return self;
}
- (void)setTeacherName:(NSString *)newValue{
_teacherName = newValue;
}
- (NSString *)teacherName{
return _teacherName;
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSString *sel = NSStringFromSelector(aSelector);
if([sel rangeOfString:@"set"].location == 0){
return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
}else{
return [NSMethodSignature signatureWithObjCTypes:"@@:"];
}
}
- (void)forwardInvocation:(NSInvocation *)anInvocation{
NSString *key = NSStringFromSelector([anInvocation selector]);
if([key rangeOfString:@"set"].location == 0){
key = [[key substringWithRange:NSMakeRange(3, key.length-4)] lowercaseString];
NSString *obj;
[anInvocation getArgument:&obj atIndex:2];//v@:@
[_propertiesDict setObject:obj forKey:key];
}else{
NSString *obj = [_propertiesDict valueForKey:[key lowercaseString]];
[anInvocation setReturnValue:&obj];
}
}
//viewController
- (void)teacher{
BRTeacher *teacher = [[BRTeacher alloc] init];
teacher.schoolName = @"北京大學(xué)";
NSLog(@"李老師所在的學(xué)校是%@",teacher.schoolName);
}
我們來(lái)分析一下程序
- 在給程序添加消息轉(zhuǎn)發(fā)功能以前,必須覆蓋兩個(gè)方法养叛,即methodSignatureForSelector:和forwardInvocation:,前一個(gè)方法的作用是為一個(gè)類的實(shí)現(xiàn)創(chuàng)建一個(gè)有效的方法簽名种呐,后者是將選擇器發(fā)一個(gè)真正實(shí)現(xiàn)了該消息的對(duì)象
- OC中的方法默認(rèn)兩個(gè)參數(shù):self和_cmd,只是被隱藏了,可以通過(guò)打印方法參數(shù)列表可以看的到一铅。self指向本身陕贮,_cmd指向方法本身堕油。
比如:- (NSString *)name 這個(gè)方法實(shí)際是有兩個(gè)參數(shù),self和_cmd
- (void)setName:(NSString *)name 這個(gè)方法實(shí)際上有三個(gè)參數(shù) self,_cmd,name - 在代碼中teacher.schoolName = @"北京大學(xué)"潘飘,程序運(yùn)行到這里會(huì)去BRTeache.m文件中尋找setSchoolName方法肮之,但是我們?cè)?m文件并沒(méi)有實(shí)現(xiàn)setter方法,于是程序就進(jìn)入methodSignatureForSelector進(jìn)行消息轉(zhuǎn)發(fā)卜录,并以"v@:@"作為方法簽名返回戈擒。這里v@:@是什么東西呢?實(shí)際上艰毒,這里的第一個(gè)字符v代表函數(shù)的返回類型是void筐高,后面三個(gè)字符參考上面2)中的解釋就可以知道,分別是self, _cmd, name這三個(gè)參數(shù)的類型id, SEL, NSString丑瞧。進(jìn)一步程序進(jìn)入forwardInvocation方法柑土,得到的key是方法名setSchoolName,利用[invocation getArgument:&obj atIndex:2]獲取我們賦給屬性的值,index=2則是因?yàn)閰?shù)的索引是2绊汹,前面已經(jīng)說(shuō)到第一個(gè)參數(shù)是self,第二個(gè)是_cmd稽屏,第三個(gè)參數(shù)才是傳的參數(shù)
- 我們?cè)倏纯慈绾潍@取屬性值,NSLog(@"李老師所在的學(xué)校是%@",teacher.schoolName);我們想打印出李老師所在的學(xué)校名稱西乖,但是.m文件并沒(méi)有實(shí)現(xiàn)getter方法狐榔,這個(gè)時(shí)候同樣會(huì)進(jìn)去消息轉(zhuǎn)發(fā)methodSignatureForSelector,以"@@:"作為簽名返回获雕,這里第一字符@代表函數(shù)返回類型NSString薄腻,第二個(gè)字符@代表self的類型id,第三個(gè)字符:代表_cmd的類型SEL届案。
- 在調(diào)試的過(guò)程中庵楷,我們發(fā)現(xiàn)了我們的teacherName的存取值沒(méi)有進(jìn)入消息轉(zhuǎn)發(fā)的方法里面,這是因?yàn)槲覀円呀?jīng)通過(guò)私有變量幫其實(shí)現(xiàn)了setter和getter方法楣颠,假設(shè)我們還有一個(gè)屬性age,標(biāo)識(shí)為@synthesize age,同樣他也不會(huì)進(jìn)入消息轉(zhuǎn)發(fā)的方法里面嫁乘,因?yàn)樗J(rèn)已經(jīng)自動(dòng)為我們加上了setter和getter方法了
以上程序分析采用[http://blog.csdn.net/haishu_zheng/article/details/12873151] 解釋的很詳細(xì)
4.@dynamic最常用的使用是在NSManagedObject中,此時(shí)不需要顯示編程setter和getter方法球碉。原因是:@dynamic告訴編譯器不做處理蜓斧,使編譯通過(guò),其getter和setter方法會(huì)在運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建睁冬,由Core Data框架為此類屬性生成存取方法挎春。