Objective-C很有趣的一個地方是唁影,它非常非常像C方篮。實際上付秕,它就是C語言加上一些其他擴展和一個運行時間(runtime)晾剖。有了這個在每個Objective-C程序中都會起作用的附加運行時間锉矢,給了它一些動態(tài)的特性。C和C++沒有運行時間齿尽,他們僅僅被編譯為完全按照代碼的順序去執(zhí)行沽损,不多也不少。Objective-C中帶有運行時間的好處是循头,它可以在你的程序運行的流程中進(jìn)行參與绵估。在Objective-C中,它包括檢查是否一個對象可以處理特定的消息卡骂,如果不能處理国裳,它就幫助你自動調(diào)用其他特定的方法去完成。initialize不是init運行時間的行為之一就是initialize全跨。雖然看起來有點像大家常見的init缝左,但是他們并不相同。在程序運行過程中螟蒸,它會在你程序中每個類調(diào)用一次initialize盒使。這個調(diào)用的時間發(fā)生在你的類接收到消息之前,但是在它的超類接收到initialize之后七嫌。舉個例子少办,比如一個叫做Duck的類:#import "Duck.h"; @implementation Duck +(void) initialize {? ? NSLog(@"Duck initialize");} -(void) init {? ? NSLog(@"Duck init");}@end我們在這里記錄initialize和init調(diào)用的時間。我們建立三個Duck對象的實例:NSLog(@"Hello, World!");Duck* duck1 = [[Duck alloc] init];Duck* duck2 = [[Duck alloc] init];Duck* duck3 = [[Duck alloc] init]; 看一下記錄:[Session started at 2008-03-23 20:03:25 -0400.]2008-03-23 20:03:25.869 initialize_example[30253:10b] Hello, World!2008-03-23 20:03:25.871 initialize_example[30253:10b] Duck initialize2008-03-23 20:03:25.872 initialize_example[30253:10b] Duck init2008-03-23 20:03:25.873 initialize_example[30253:10b] Duck init2008-03-23 20:03:25.873 initialize_example[30253:10b] Duck init我們可以看到诵原,雖然我們創(chuàng)建了3個Duck的實例英妓,但是initialize僅僅被調(diào)用了一次。我們也可以看到绍赛,直到我們創(chuàng)建了一個Duck的實例蔓纠,initialize才被調(diào)用。但是如果Duck有一個子類的話吗蚌,比如我們建一個Duck的子類叫做Chicken(好怪異……)#import#import "Duck.h"
@interface Chicken : Duck {
}
@end
注意Chicken這個類并沒有實現(xiàn)initialize方法腿倚。
如果我們同樣運行這個程序,但是加上一個Chicken的實例:
NSLog(@"Hello, World!");
Duck* duck1 = [[Duck alloc] init];
Duck* duck2 = [[Duck alloc] init];
Duck* duck3 = [[Duck alloc] init];
Chicken* chicken = [[Chicken alloc] init];
我們期待看到4個Duck的init調(diào)用(因為我們建立了3個Duck和一個Chicken)蚯妇,但是我們看到了這樣情況:
[Session started at 2008-03-23 20:13:34 -0400.]
2008-03-23 20:13:34.696 initialize_example[30408:10b] Hello, World!
2008-03-23 20:13:34.698 initialize_example[30408:10b] Duck initialize
2008-03-23 20:13:34.699 initialize_example[30408:10b] Duck init
2008-03-23 20:13:34.700 initialize_example[30408:10b] Duck init
2008-03-23 20:13:34.700 initialize_example[30408:10b] Duck init
2008-03-23 20:13:34.700 initialize_example[30408:10b] Duck initialize
2008-03-23 20:13:34.701 initialize_example[30408:10b] Duck init
我們看到了4個Duck的init和2個Duck的initialize方法敷燎。這是怎么回事呢暂筝?
看來如果一個子類沒有實現(xiàn)initialize方法,那么超類會調(diào)用這個方法兩次硬贯,一次為自己焕襟,而一次為子類。
我們在Duck的initialize類中記錄一下類名饭豹,這樣可以看得更清楚:
+(void) initialize {
NSLog(@"Duck initialize class:%@",[self class]);
}
現(xiàn)在看明白了:
[Session started at 2008-03-23 20:21:08 -0400.]
2008-03-23 20:21:08.816 initialize_example[30513:10b] Hello, World!
2008-03-23 20:21:08.818 initialize_example[30513:10b] Duck initialize class:Duck
2008-03-23 20:21:08.819 initialize_example[30513:10b] Duck init
2008-03-23 20:21:08.820 initialize_example[30513:10b] Duck init
2008-03-23 20:21:08.820 initialize_example[30513:10b] Duck init
2008-03-23 20:21:08.820 initialize_example[30513:10b] Duck initialize class:Chicken
2008-03-23 20:21:08.821 initialize_example[30513:10b] Duck init
如果你希望確定只用了initialize一次用來實現(xiàn)某些單獨運行的工作鸵赖,或者希望實現(xiàn)僅僅運行一次的方法,檢查一下[self class]拄衰,才能確定是否是你希望做到的效果它褪。