Objective-C 中關(guān)于self.用法
Objective-C 中關(guān)于self.用法是本文要講述的內(nèi)容,不多說(shuō),直接進(jìn)入話(huà)題,我們經(jīng)常會(huì)在官方文檔里看到這樣的代碼:
- MyClass.h
 - [/lang]
 - @interface MyClass : NSObject {
 - MyObject *myObject;
 - }
 - @property (nonatomic, retain) MyObject *myObject;
 - @end
 - MyClass.m
 - @synthesize myObject;
 - -(id)init{
 - if(self = [super init]){
 - MyObject * aMyObject = [[MyObject alloc] init];
 - self.myObject = aMyObject;
 - [aMyObject release];
 - }
 - return self;
 - }
 
有人就問(wèn), 為什么要這么復(fù)雜的賦值? 為什么要加self. ? 直接寫(xiě)成self.myObject = [[MyObject alloc] init];不是也沒(méi)有錯(cuò)么? 不加self有時(shí)好像也是正常的? 現(xiàn)在我們來(lái)看看內(nèi)存管理的內(nèi)容:
先看間接賦值的:
1.加self.:
- MyObject * aMyObject = [[MyObject alloc] init]; //aMyObject retainCount = 1;
 - self.myObject = aMyObject; //myObject retainCount = 2;
 - [aMyObject release];//myObject retainCount = 1;
 
2. 不加self.:
- MyObject * aMyObject = [[MyObject alloc] init]; //aMyObject retainCount = 1;
 - myObject = aMyObject; //myObject retainCount = 1;
 - [aMyObject release];//對(duì)象己經(jīng)被釋放
 
再看直接賦值的:
3.加self.:
- self
 - .myObject
 - = [[
 - MyObject
 - alloc] init]
 - ;
 - //myObject retainCount = 2;
 
4. 不加self.:
- myObject = [[MyObject alloc] init]; //myObject retainCount = 1;
 
現(xiàn)在是不是有點(diǎn)暈, 我們先來(lái)把代碼改一下, 官方的一種常見(jiàn)寫(xiě)法:
- MyClass.h
 - @interface MyClass : NSObject {
 - MyObject * _myObject;
 - }
 - @property (nonatomic, retain) MyObject *myObject;
 - @end
 - MyClass.m
 - @synthesize myObject = _myObject;
 
OK, 你現(xiàn)在再試下, 如果你用self._myObject = aMyObject; 或者 myObject = aMyObject; 你會(huì)得到一個(gè)錯(cuò)誤, 為什么呢, 這里就是和Obj-c的存取方法有關(guān)了. 說(shuō)白了很簡(jiǎn)單 , 大家都知道, @property (nonatomic, retain) MyObject *myObject; 是為一個(gè)屬性設(shè)置存取方法, 只是平時(shí)我們用的方法名和屬性名是一樣的,現(xiàn)在你把它寫(xiě)成不同的名字, 就會(huì)很清楚了. _myObject是屬性本身, myObject是存取方法名.
現(xiàn)在我們知道self.是訪問(wèn)屬性的存取方法了, 那存取方法又怎么工作的? self.myObject = [[MyObject alloc] init]; 為什么會(huì)有內(nèi)存泄露?
關(guān)于nonatomic我不多解釋了, 它不是我要講的重點(diǎn), 而且我也沒(méi)完全搞清楚, 不誤導(dǎo)大家. 我只說(shuō)assign, retain ,copy.
get方法是:
- -(MyObject*)myObject{
 - return _myObject;
 - }
 
Set方法是:
- // assign
 - -(void)setMyObject:(id)newValue{
 - _myObject = newValue;
 - }
 - // retain
 - -(void)setMyObject:(id)newValue{
 - if (_myObject != newValue) {
 - [_myObject release];
 - _myObject = [newValue retain];
 - }
 - }
 - // copy
 - -(void)setMyObject:(id)newValue{
 - if (_myObject != newValue) {
 - [_myObject release];
 - _myObject = [newValue copy];
 - }
 - }
 
其實(shí)這些方法里還有別的內(nèi)容, 并不只是這些. 而且這些方法可以被重寫(xiě). 比如你寫(xiě)一個(gè)
- -(MyObject*)myObject{
 - return _myObject;
 - }
 
放在你的類(lèi)里, 你調(diào)用self.myObject時(shí)(不要把它放在等號(hào)左邊, 那會(huì)調(diào)用get方法)就會(huì)調(diào)用這個(gè)方法.
這里多說(shuō)一句, @property 是為你設(shè)置存取方法, 和你的屬性無(wú)關(guān), 你可以只寫(xiě)一句
- @property (readonly) NSString *name;
 
在你的類(lèi)里實(shí)現(xiàn)
- -(NSString*)name{
 - NSLog(@"name");
 - return @"MyClass";
 - }
 
同樣可以用self.name調(diào)用.
現(xiàn)在回頭 說(shuō)說(shuō)我們開(kāi)始的那四個(gè)賦值, 當(dāng)不用self.的時(shí)候, 那句話(huà)只是一般的賦值, 把一個(gè)指針賦給另一個(gè)指針, 不會(huì)對(duì)分配的內(nèi)存有任何影響, 所以2中不要最后[aMyObject release];這句話(huà)和4是一回事. 這里就不多說(shuō)了.我們看看1和3,
當(dāng)調(diào)用 setMyObject:方法時(shí), 對(duì)newValue 做了一次retain操作, 我們必須把原來(lái)的newValue釋放掉, 不然就會(huì)內(nèi)存泄露, 在1里, 我們有個(gè)aMyObject可以用來(lái)釋放, 在3里, 我們無(wú)法釋放它, 所以, 在3里, 我們會(huì)多出來(lái)一個(gè)retainCount. 內(nèi)存泄露了.
說(shuō)了這么多, 我只想讓大家清楚, 什么是調(diào)用屬性本身, 什么是調(diào)用存取方法. 怎么樣才能避免內(nèi)存泄露, 而且, 以上例子里是在自己類(lèi)里的調(diào)用, 如果這個(gè)類(lèi)被別的類(lèi)調(diào)用時(shí), 更要注意一些,
順便說(shuō)一下, 如果你想在其它類(lèi)訪問(wèn)對(duì)象屬性, 而不是通過(guò)存取方法, 你可以用myClass -> myObject來(lái)訪問(wèn), 這樣是直接訪問(wèn)對(duì)象本身, 不過(guò)你先要把myObject設(shè)成@public. 但這個(gè)是官方不提倡的,
代碼比較簡(jiǎn)單, 我還是發(fā)出來(lái), 高人們可以忽略了 附件: SelfExample.zip (18 K)http://www.cocoachina.com/bbs/job.php?action=download&aid=6639
小結(jié):Objective-C 中關(guān)于self.用法的一些總結(jié)的內(nèi)容介紹完了,希望本文對(duì)你有所幫助!
帖子地址 http://www.cocoachina.com/bbs/read.php?tid-12850-fpage-11-page-1.html,歡迎參與討論。















 
 
 
 
 
 
 