@synthesize與@dynamic的區(qū)別

1.@property有兩個(gè)對(duì)應(yīng)的詞,一個(gè)是 @synthesize,一個(gè)是 @dynamic。如果 @synthesize和 @dynamic都沒寫,那么默認(rèn)的就是@syntheszie var = _var,也就是自動(dòng)給我們生成成員變量,但如果我們同時(shí)手動(dòng)實(shí)現(xiàn)了setter和getter方法,property就不會(huì)自動(dòng)生成成員變量,其他的情況下會(huì)自動(dòng)生成成員變量

A9A0F360-2FE6-48A4-85EF-B7CF6355F136.png

2.@synthesize的語義是如果你沒有手動(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ì)去訪問_age這個(gè)成員變量,如果找不到_age這個(gè)成員變量,會(huì)自動(dòng)生成一個(gè)叫做_age的私有成員變量。如果在.m文件中同時(shí)實(shí)現(xiàn)getter和setter的時(shí)候需要在.m文件中定義@synthesize age = _age

07664644-FC5A-4D56-BB50-0357E81F2010.png

3.@dynamic告訴編譯器,屬性的setter和getter方法由用戶自己實(shí)現(xiàn),不自動(dòng)生成,(當(dāng)然對(duì)于readonly的屬性只需要提供getter方法即可)。假如一個(gè)屬性被聲明為@dynamic var ,然后你沒有提供@setter和getter方法,編譯的時(shí)候沒問題,但是當(dāng)程序運(yùn)行到instanve.var = someVar,由于我們沒有實(shí)現(xiàn)setter方法,程序會(huì)崩潰,或者當(dāng)運(yùn)行到someVar = var時(shí),由于缺getter方法同樣會(huì)導(dǎo)致程序崩潰,編譯時(shí)沒問題,運(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)建并綁定,我們來看一下代碼如何實(shí)現(xiàn)

通過私有變量來實(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;
}

通過消息轉(zhuǎn)發(fā)來實(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);
}

我們來分析一下程序

  • 在給程序添加消息轉(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,只是被隱藏了,可以通過打印方法參數(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文件并沒有實(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)說到第一個(gè)參數(shù)是self,第二個(gè)是_cmd,第三個(gè)參數(shù)才是傳的參數(shù)
  • 我們?cè)倏纯慈绾潍@取屬性值,NSLog(@"李老師所在的學(xué)校是%@",teacher.schoolName);我們想打印出李老師所在的學(xué)校名稱,但是.m文件并沒有實(shí)現(xiàn)getter方法,這個(gè)時(shí)候同樣會(huì)進(jìn)去消息轉(zhuǎn)發(fā)methodSignatureForSelector,以"@@:"作為簽名返回,這里第一字符@代表函數(shù)返回類型NSString,第二個(gè)字符@代表self的類型id,第三個(gè)字符:代表_cmd的類型SEL。
  • 在調(diào)試的過程中,我們發(fā)現(xiàn)了我們的teacherName的存取值沒有進(jìn)入消息轉(zhuǎn)發(fā)的方法里面,這是因?yàn)槲覀円呀?jīng)通過私有變量幫其實(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告訴編譯器不做處理,使編譯通過,其getter和setter方法會(huì)在運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建,由Core Data框架為此類屬性生成存取方法。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • 我和婆婆之間總會(huì)有一些不愉快的磨合,在干家務(wù)活上如洗衣服,擦房間時(shí)她總愛邊上指手劃腳,衣服曬的不整齊啦,床的靠背要...
    浩莉閱讀 185評(píng)論 2 4
  • 關(guān)于床上用品的那些困惑: 迷思之一:總覺得美劇里面別人的床看上去很舒適。可是自己只能換類似的款式卻達(dá)不到那樣舒適的...
    有櫝閱讀 581評(píng)論 0 1
  • 突然翻到很久以前的畫本,丟之前先拍照留個(gè)紀(jì)念,現(xiàn)在看來畫的真是差,線條雜亂無章
    林小北x2閱讀 283評(píng)論 2 1
  • 農(nóng)村人樸實(shí)善良者居多,因此,農(nóng)村絕大時(shí)候都是呈現(xiàn)著寧?kù)o而祥和的氛圍。唯有雞犬之聲相聞,屋脊炊煙裊裊。 偌大一個(gè)村子...
    泰山寒梅閱讀 945評(píng)論 33 28