Objective-C之@property和@synthesize
我用了不到一周的時間學習了Objective- C,后面的大部分時間我都在了解如何使用iOS的SDK和一些高級的話題,到目前已經(jīng)有兩個多月的時間了。目前能做一些簡單的應用,但是在寫代碼的時候明 顯感覺到基礎不夠扎實,畢竟一周的時間只能對一門語言有個概覽。要想精通一門語言是遠遠不夠的。
所以我把自己學習過程中遇到的一些問題整理在博客上,這也是一個學習理解的過程。
今天要說的內(nèi)容是Objective-C 中的 @property和@synthesize。在這之前先講講訪問器(Accessor),也就是我們所知道的setter和getter方法。 《Cocoa Design Patterns》中的將它歸類為基礎模式中的一種。訪問器是很重要的技術(shù),用來訪問和設置對象的實例變量(不是指對象本身,而是對象中的屬性)。有時候 可能需要用不同的方式或者通過計算等方式來獲取或設置實例變量,訪問器給了我們很大的靈活性。在Cocoa中訪問器有很多的優(yōu)點:
- 實現(xiàn)靈活性。 可以在訪問器中改變并實現(xiàn)不同的實例變量訪問方式而不影響其他代碼。
- 可維護性。通過訪問器對實例變量的更改易于維護。
- 內(nèi)存管理。訪問器方法提供了簡單的方法去遵守Cocoa的約定把內(nèi)存管理代碼隔離在少部分代碼中。
- 支持KVC和KVO。 KVC和KVO是很強大的技術(shù)。但是它們依賴于正確命名訪問器。
下面這段代碼簡單的實現(xiàn)了一個訪問器(setter和getter):
- //setter
- -(void)setStuName:(NSString *)stuName
- { //_stuName 是實例變量
- if (_stuName != stuName)
- {
- [_stuName release];
- _stuName = [stuName copy];
- }
- }
- //getter
- -(NSString *)stuName
- {
- return _stuName;
- }
上面代碼中的setter中還涉及到一定的內(nèi)存管理,既然這個技術(shù)這么重要,那么有沒有一種更方便的方法去做呢?答案就是@property和@synthesize。它們是Objective-C 2.0加入的指令,前者用于聲明,后者用于合成訪問器,結(jié)合使用就可以自動生成訪問器了。
下面這段代碼使用@property和@synthesize:
- @interface Student : NSObject
- @property (nonatomic, copy) NSString *stuName;
- @end
- @implementation Student
- @synthesize stuName = _stuName;
- @end
這段代碼的效果跟上面代碼的效果是一樣的,是不是很方便呢?
使用@property和@synthesize很方便,但又給我們帶來了很多疑問比如在上面的代碼中又出現(xiàn)了nonatomic和copy,是什么意 思?在@property中還有其他幾個關鍵字,它們都是有特殊作用的,我把它們分為三類分別是:原子性,訪問器控制,內(nèi)存管理。
原子性
atomic(默認):atomic意為操作是原子的,意味著只有一個線程訪問實例變量。atomic是線程安全的至少在當前的訪器上我是安全的。它是一個默認的,但是很少使用。它的比較慢,這跟ARM平臺和內(nèi)部鎖機制有關。
nonatomic: nonatomic跟atomic剛好相反。表示非原子的,可以被多個線程訪問。它的速度比atomic快。但不能保證在多線程環(huán)境下的安全性,在單線程和明確只有一個線程訪問的情況下廣泛使用。
訪問器控制
readwrite(默認):readwrite是默認的,表示同時擁有setter和getter。
readonly: readonly 表示只有g(shù)etter沒有setter。
有時候為了語意更明確可能需要自定義訪問器的名字:
- @property (nonatomic, setter = mySetter:,getter = myGetter ) NSString *name;
最常見的是BOOL類型,比如標識View是否隱藏的屬性hidden??梢赃@樣聲明
- @property (nonatomic,getter = isHidden ) BOOL hidden;
要注意修改setter或者getter的名字是存在副作用的,可能會使KVC和KVO無法正常工作。
內(nèi)存管理
retain:使用了retain意味著實例變量要獲取傳入?yún)?shù)的所有權(quán)。具體表現(xiàn)在setter中對實例變量先release然后將參數(shù) retain之后傳給它。下面這段代碼展示了retain類似的行為:
- -(void)setStuName:(NSString *)stuName
- {
- if (_stuName != stuName)
- {
- [_stuName release];
- _stuName = [stuName retain];
- }
- }
assign(默認):用于值類型,如int、float、double和NSInteger,CGFloat等表示單純的復制。還包括不存在所有權(quán)關系的對象,比如常見的delegate。
strong:是在ARC伴隨IOS引入的時候引入的關鍵字是retain的一個可選的替代。表示實例變量對傳入的參數(shù)要有所有權(quán)關系即強引用。strong跟retain的意思相同并產(chǎn)生相同的代碼,但是語意上更好更能體現(xiàn)對象的關系。
weak: weak跟assign的效果相似,不同的是weak在對象被回收之后自動設置為nil。而且weak智能用在iOS 5或以后的版本,對于之前的版本,使用unsafe_unretained。
unsafe_unretained:weak的低版本替代。
copy:copy是為是實例變量保留一個自己的副本。
現(xiàn)在明白了@property是怎么回事了,但是@synthesize是怎么回事,看看之前的***段代碼:
- @synthesize stuName = _stuName;
這里的stuName = _stuName是什么意思?stuName是propertyName跟@property聲明的名字一樣。而后面的_stuName 是實例變量名。生成的訪問器就是來訪問的 _stuName的。代碼的樣子就和最開始那setter和getter代碼所描述的一樣。
注意一個問題,我們并沒有聲明_stuName這個變量,這是編譯器自動幫我們創(chuàng)建的。 如果這段指令我換個寫法:@synthesize stuName = a; 并且我們沒有在interface里面聲明這個變量,那么會自動創(chuàng)建一個變量a。
如果這里寫成這樣:
- <em>@synthesize stuName;
- //等同于
- @synthesize stuName = stuName;</em>
在Xcode4.4中,Xcode添加的一些新的編譯特性。其中一個就是默認合成(Default Synthesis)。默認合成就不再需要顯示的使用@synthesize指令了,這很方便但是要注意的是,默認合成遵守的約定,這里的也就是命名規(guī)則是propertyName = _propertyName。
下面一段代碼幫助理解:
- //對于下面的@propety
- @property (nonatomic, copy) NSString *stuName;
- //默認合成的規(guī)則是這樣:
- @synthesize stuName = _stuName;
以上是我所了解的@property和@synthesize,如果跟你的理解不同,或者有什么錯誤,請給我留言:)。