成员变量和属性区别(@property那点事儿)
歷史由來:
接觸iOS的人都知道,@property聲明的屬性默認(rèn)會生成一個_類型的成員變量,同時也會生成setter/getter方法。?
但這只是在iOS5之后,蘋果推出的一個新機(jī)制??蠢洗a時,經(jīng)常看到一個大括號里面定義了成員變量,同時用了@property聲明,而且還在@implementation中使用@synthesize方法。?
如下:
其實(shí),發(fā)生這種狀況根本原因是蘋果將默認(rèn)編譯器從GCC轉(zhuǎn)換為LLVM(low level virtual machine),才不再需要為屬性聲明實(shí)例變量了。
在沒有更改之前,屬性的正常寫法需要成員變量+ @property + @synthesize 成員變量三個步驟。?
如果我們只寫成員變量+ @property:
但更換為LLVM之后,編譯器在編譯過程中發(fā)現(xiàn)沒有新的實(shí)例變量后,就會生成一個下劃線開頭的實(shí)例變量。因此現(xiàn)在我們不必在聲明一個實(shí)例變量。(注意:==是不必要,不是不可以==)?
當(dāng)然我們也熟知,@property聲明的屬性不僅僅默認(rèn)給我們生成一個_類型的成員變量,同時也會生成setter/getter方法。
在.m文件中,編譯器也會自動的生成一個實(shí)例變量_myString。那么在.m文件中可以直接的使用_myString實(shí)例變量,也可以通過屬性self.myString.都是一樣的。
注意這里的self.myString其實(shí)是調(diào)用的myString屬性的setter/getter方法。這與C++中點(diǎn)的使用是有區(qū)別的,C++中的點(diǎn)可以直接訪問成員變量(也就是實(shí)例變量)。
例如在oc中有如下代碼
@interface MyViewController :UIViewController {NSString *name; } @end?
在這段代碼里面只是聲明了一個成員變量,并沒有setter/getter方法。所以訪問成員變量時,可以直接訪問name,也可以像C++一樣用self->name來訪問,但絕對不能用self.name來訪問。
- 擴(kuò)展:很多人覺得OC中的點(diǎn)語法比較奇怪,實(shí)際是OC設(shè)計(jì)人員有意為之。
- 點(diǎn)表達(dá)式(.)看起來與C語言中的結(jié)構(gòu)體訪問以及java語言匯總的對象訪問有點(diǎn)類似,如果點(diǎn)表達(dá)式出現(xiàn)在等號?=?左邊,調(diào)用該屬性名稱的setter方法。如果點(diǎn)表達(dá)式出現(xiàn)在=右邊,調(diào)用該屬性名稱的getter方法。
- OC中點(diǎn)表達(dá)式(.)其實(shí)就是調(diào)用對象的setter和getter方法的一種快捷方式,self.myString = @"張三";實(shí)際就是[self setmyString:@"張三"];
首先我們要明白,@synthesize?生成了setter/getter方法。?
雖然現(xiàn)在直接使用@property時,編譯器會自動為你生成以下劃線開頭的實(shí)例變量_myString,不需要自己手動再去寫實(shí)例變量。而且也不在.m文件中通過@synthesize myString;生成setter/getter方法。但在看老代碼的時候,我們依舊可以看到有人使用成員變量+ @synthesize 成員變量的形式。
那么問題來了:?
我們能否認(rèn)為新編譯器LLVM下的@property?== 老編譯器GCC的?成員變量+ @property + @synthesize 成員變量呢?
答案是否定的。?
因?yàn)槌蓡T變量+ @property + @synthesize 成員變量的形式,編譯器不會幫我們生成_成員變量,因此不會操作_成員變量了;?
同時@synthesize?還有一個作用,可以指定與屬性對應(yīng)的實(shí)例變量,?
例如@synthesize myString = xxx;?
那么self.myString其實(shí)是操作的實(shí)例變量xxx,而非_String了。?
?
三、類別中的屬性property
類與類別中添加的屬性要區(qū)分開來,因?yàn)轭悇e中只能添加方法,不能添加實(shí)例變量。經(jīng)常會在ios的代碼中看到在類別中添加屬性,這種情況下,是不會自動生成實(shí)例變量的。比如在:UINavigationController.h文件中會對UIViewController類進(jìn)行擴(kuò)展
@interface UIViewController (UINavigationControllerItem) @property(nonatomic,readonly,retain) UINavigationItem *navigationItem; @property(nonatomic) BOOL hidesBottomBarWhenPushed; @property(nonatomic,readonly,retain) UINavigationController *navigationController; @end這里添加的屬性,不會自動生成實(shí)例變量,這里添加的屬性其實(shí)是添加的getter與setter方法。
注意一點(diǎn),匿名類別(匿名擴(kuò)展)是可以添加實(shí)例變量的,非匿名類別是不能添加實(shí)例變量的,只能添加方法,或者屬性(其實(shí)也是方法)。
四、成員變量、實(shí)例變量、屬性變量的聯(lián)系
@interface MyViewController :UIViewControlle
{
UIButton *yourButton;
int count;
id data;
}
@property (nonatomic, strong) UIButton *myButton;
@end
? 在{?? } 中所聲明的變量都為成員變量。 所以yourButton、count、data都是成員變量。既然如此,實(shí)例變量又是什么意思呢?
實(shí)例變量本質(zhì)上就是成員變量,只是實(shí)例是針對類而言,實(shí)例是指類的聲明。{?? }中的yourButton就是實(shí)例變量。id 是OC特有的類,本質(zhì)上講id等同于(void *)。所以id data屬于實(shí)例變量。
成員變量用于類內(nèi)部,無需與外界接觸的變量。因?yàn)槌蓡T變量不會生成set、get方法,所以外界無法與成員變量接觸。根據(jù)成員變量的私有性,為了方便訪問,所以就有了屬性變量。屬性變量的好處就是允許讓其他對象訪問到該變量(因?yàn)閷傩詣?chuàng)建過程中自動產(chǎn)生了set 和get方法)。當(dāng)然,你可以設(shè)置只讀或者可寫等,設(shè)置方法也可自定義。所以,屬性變量是用于與其他對象交互的變量。
綜上所述可知:成員變量是定義在{}號中的變量,如果變量的數(shù)據(jù)類型是一個類則稱這個變量為實(shí)例變量。因?yàn)閷?shí)例變量是成員變量的一種特殊情況,所以實(shí)例變量也是類內(nèi)部使用的,無需與外部接觸的變量,這個也就是所謂的類私有變量。而屬性變量是用于與其他對象交互的變量。
但是,現(xiàn)在大家似乎都不怎么喜歡用成員變量來定義類的變量,都喜歡用屬性變量來定義類的變量。把需要與外部接觸的變量定義在.h文件中,只在本類中使用的變量定義在.m文件中。
?
首先來區(qū)別一下實(shí)例變量、成員變量的區(qū)別:可以看到在接口 @interface 括號里面的統(tǒng)稱為”成員變量”,實(shí)例變量是成員變量中的一種! 實(shí)例變量的英文翻譯是?Instance?Variable?(object-specific storage)? 實(shí)例的英文翻譯為Instance(manifestation? of a? class)? 說的是“類的表現(xiàn)”,說明實(shí)例變量應(yīng)該是由類定義的變量! 除去基本數(shù)據(jù)類型int float ....等,其他類型的變量都叫做實(shí)例變量。 **實(shí)例變量+基本數(shù)據(jù)類型變量=成員變量** 接下來說說屬性: ? 在@property(描述1 , 描述2 , 描述3)(class *) varName?里面,有3個描述詞需要填寫(也可以不填寫取默認(rèn)值) 1. nonatomic<-->atomic 2. readwrite<-->readonly 3. retain/copy/assign ? 首先來介紹一下: retain:他指的是將某個內(nèi)存區(qū)域的指針賦值給變量,同時把該內(nèi)存區(qū)域的引用計(jì)數(shù)器加1.每執(zhí)行一次,該內(nèi)存區(qū)域的引用計(jì)數(shù)器就要加1,當(dāng)該區(qū)域的引用計(jì)數(shù)器變?yōu)?的時候內(nèi)存區(qū)域被釋放! ? copy:它指的是將目標(biāo)內(nèi)存區(qū)域的值復(fù)制一份,然后開辟新的內(nèi)存區(qū)域(新的指針)粘貼這個值。同時變量被賦值為新內(nèi)存區(qū)域的指針! ? assign:它指的是,僅只把目標(biāo)內(nèi)存區(qū)域的指針賦值給變量,該內(nèi)存區(qū)域的引用計(jì)數(shù)器不發(fā)生變化! ? 1、2兩點(diǎn)不做解釋,3中的retain、copy、assign都是指的,在自動生成setter函數(shù)的時候,編譯器需要識別個描述詞來生成對應(yīng)的setter函數(shù)!需要注意的是,如果沒有加上該類的描述詞,系統(tǒng)默認(rèn)該變量的setter方法采取assign的方式。 在頭文件中.h一般在{}里面會有定義的實(shí)例變量?? 示例: .h @property (automic,retain) NSString * abc; .m @sythesize abc;
//在寫了@sythesize abc;的情況下,系統(tǒng)不會自動生成實(shí)例變量“_abc”,直接通過變量名abc?,也就是直接使用變量名在賦值運(yùn)算的時候(=號左邊),只是將內(nèi)存區(qū)域的指針賦值給變量,相當(dāng)于assgin. ??如果是通過“點(diǎn)語句”self.abc= 來賦值,就要看在@property中定義的是copy、retain、assign哪一種了,如果沒有加上上述描述詞,就默認(rèn)為assign。
//如果沒有寫@sythesize abc; ?系統(tǒng)會默認(rèn)自動在.h文件{}中添加一個 不可見的 加“_”的成員變量(即使是變量名中本身就帶有“_”)
//括號里面定義的都是成員變量(基本數(shù)據(jù)類型+類生成變量),里面的變量可以在.m文件中通過“變量名稱”、self->“變量名稱”直接訪問到括號里面的變量,但是,這樣的賦值訪問只能是assign,原對象的引用計(jì)數(shù)器不會發(fā)生變化。
//1.@sythesize 變量名;2.@sythesize 變量名=_變量名;3.不寫@sythesize (一下提到的變量名都是指的在頭文件中@property 中定義的變量)
1.成員變量,實(shí)例變量通過“變量名”或者self->“變量名”直接訪問到,賦值(assign)。self.變量名 實(shí)現(xiàn)setter,getter方法。
2.成員變量,實(shí)例變量通過“_變量名”或者self->“_變量名”直接訪問到,賦值(assign)。self.變量名 實(shí)現(xiàn)setter,getter方法。
3.成員變量。實(shí)例變量(系統(tǒng)自動在原來變量名前加上“_”來生成的實(shí)例,成員變量),直接通過self->_變量名,或者變量名直接訪問到(assign)。self.變量名 實(shí)現(xiàn)setter,getter方法。
?
如果在頭文件中沒有通過@property定義的變量,但是在{}中有定義成員變量,在實(shí)現(xiàn)文件中也也沒有@sythesize ,那么可以直接通過self->“{}中的變量名”,或者直接使用“{}中的變量名”來訪問賦值,這樣的變量沒有定義setter,getter函數(shù),只能是assign的方式賦值。
?
//再來分析一下@sythesize中的寫法,@sythesize abc 直接在.m文件中使用self.abc可以調(diào)用成員變量的setter、getter函數(shù),直接調(diào)用成員變量名稱abc就為訪問該變量的指針,對成員變量直接賦值等同于ASSIGN效果。
參考: http://blog.csdn.net/u012946824/article/details/51788565 http://www.cnblogs.com/AnnieBabygn/p/5335350.html?
轉(zhuǎn)載于:https://www.cnblogs.com/huangzs/p/7508583.html
總結(jié)
以上是生活随笔為你收集整理的成员变量和属性区别(@property那点事儿)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: git中的暂存区与工作区
- 下一篇: [js高手之路]深入浅出webpack教