因?yàn)?Cocoa 開發(fā)環(huán)境已經(jīng)在新建一個(gè)項(xiàng)目時(shí)幫助我們進(jìn)行很多配置,這導(dǎo)致了不少剛接觸 iOS 的開發(fā)者都存在基礎(chǔ)比較薄弱的問題,其中一個(gè)最顯著的現(xiàn)象就是很多人無法說清一個(gè) app 啟動的流程驾凶。程序到底是怎么開始的,AppDelegate 到底是什么炬藤,xib 或者 storyboard 是怎么被加載到屏幕上的晾捏?這一系列的問題雖然在開發(fā)中我們不會每次都去關(guān)心和自己配置,但是如果能進(jìn)行一些了解的話對于程序各個(gè)部分的職責(zé)的明確會很有幫助穿撮。
在 C 系語言中缺脉,程序的入口都是 main 函數(shù)痪欲。對于一個(gè) Objective-C 的 iOS app 項(xiàng)目,在新建項(xiàng)目時(shí)攻礼, Xcode 將幫我們準(zhǔn)備好一個(gè) main.m 文件业踢,其中就有這個(gè) main 函數(shù):
int main(int argc, char * argv[]) {
? ? ? ? ? ? ? ? @autoreleasepool {
? ? ? ? ? ? ? ? ? ?return UIApplicationMain(argc, argv, nil,NSStringFromClass([AppDelegate class]));
? ? ? ? ? ? ? ? }
}
在這里我們調(diào)用了 UIKit 的 UIApplicationMain 方法。這個(gè)方法將根據(jù)第三個(gè)參數(shù)初始化一個(gè) UIApplication 或其子類的對象并開始接收事件 (在這個(gè)例子中傳入 nil礁扮,意味使用默認(rèn)的 UIApplication)陨亡。最后一個(gè)參數(shù)指定了 AppDelegate 類作為應(yīng)用的委托,它被用來接收類似 didFinishLaunching 或者 didEnterBackground 這樣的與應(yīng)用生命周期相關(guān)的委托方法深员。另外负蠕,雖然這個(gè)方法標(biāo)明為返回一個(gè) int,但是其實(shí)它并不會真正返回倦畅。它會一直存在于內(nèi)存中遮糖,直到用戶或者系統(tǒng)將其強(qiáng)制終止。
了解了這些后叠赐,我們就可以來看看 Swift 的項(xiàng)目中對應(yīng)的情況了欲账。新建一個(gè) Swift 的 iOS app 項(xiàng)目后,我們會發(fā)現(xiàn)所有文件中都沒有一個(gè)像 Objective-C 時(shí)那樣的 main 文件芭概,也不存在 main 函數(shù)赛不。唯一和 main 有關(guān)系的是在默認(rèn)的 AppDelegate 類的聲明上方有一個(gè) @UIApplicationMain 的標(biāo)簽。
不說可能您也已經(jīng)猜到罢洲,這個(gè)標(biāo)簽做的事情就是將被標(biāo)注的類作為委托踢故,去創(chuàng)建一個(gè) UIApplication 并啟動整個(gè)程序。在編譯的時(shí)候惹苗,編譯器將尋找這個(gè)標(biāo)記的類殿较,并自動插入像 main 函數(shù)這樣的模板代碼。我們可以試試看把 @UIApplicationMain 去掉會怎么樣:
Undefined symbols _main
說明找不到 main 函數(shù)了桩蓉。
在一般情況下淋纲,我們并不需要對這個(gè)標(biāo)簽做任何修改,但是當(dāng)我們?nèi)绻胍褂?UIApplication 的子類而不是它本身的話院究,我們就需要對這部分內(nèi)容 “做點(diǎn)手腳” 了洽瞬。
剛才說到,其實(shí) Swift 的 app 也是需要 main 函數(shù)的业汰,只不過默認(rèn)情況下是 @UIApplicationMain 幫助我們自動生成了而已伙窃。和 C 系語言的 main.c 或者 main.m 文件一樣,Swift 項(xiàng)目也可以有一個(gè)名為 main.swift 特殊的文件蔬胯。在這個(gè)文件中对供,我們不需要定義作用域,而可以直接書寫代碼。這個(gè)文件中的代碼將作為 main 函數(shù)來執(zhí)行产场。比如我們在刪除 @UIApplicationMain 后鹅髓,在項(xiàng)目中添加一個(gè) main.swift 文件,然后加上這樣的代碼:
UIApplicationMain(C_ARGC, C_ARGV, nil,NSStringFromClass(AppDelegate))
現(xiàn)在編譯運(yùn)行京景,就不會再出現(xiàn)錯(cuò)誤了窿冯。當(dāng)然,我們還可以通過將第三個(gè)參數(shù)替換成自己的 UIApplication 子類确徙,這樣我們就可以輕易地做一些控制整個(gè)應(yīng)用行為的事情了醒串。比如將 main.swift 的內(nèi)容換成:
import UIKit
class MyApplication: UIApplication {
? ? ? ? ? ?override func sendEvent(event: UIEvent) {
? ? ? ? ? ? ? ? ? super.sendEvent(event)
? ? ? ? ? ? ? ? ?println("Event sent: \(event)");
? ? ? ? }
}
UIApplicationMain(C_ARGC,C_ARGV,NSStringFromClass(MyApplication), NSStringFromClass(AppDelegate))
這樣每次發(fā)送事件 (比如點(diǎn)擊按鈕) 時(shí),我們都可以監(jiān)聽到這個(gè)事件了鄙皇。