Xcode4.2以前的main函數以下:app
int main(int argc, char *argv[])ide
{函數
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];oop
int retVal = UIApplicationMain(argc, argv, nil, nil);google
[pool release];url
return retVal;spa
}.net
Xcode4.2工程中的主函數爲代理
int main(int argc, char *argv[])code
{
return UIApplicationMain(argc, argv, nil, NSStringFromClass([TCAppDelegate class]));
}
}
能夠看出一個重要的變化是在4.2使用了ARC技術後,NSAutoreleasePool被廢棄,改用@autoreleasepool ,這裏請不要該回原先的方式,若是改變後,在開啓ARC選項後,程序將不能經過編譯。
不論那個版本,UIApplicationMain函數都是程序的關鍵點,下面是對這個函數的分析:
UIApplicationMain()函數是初始化程序的核心,它接受4個參數。
argc和argv:來自於main()接受的兩個參數;
第三個參數:主要類(principal class),必須是UIApplication或其子類的名字,它表明着當前iPhone程序自己,這個程序會去讀info.plist文件獲取配置信息,其中包括主nib文件的值,通常爲MainWindow(.xib);若是該參數爲nil,則默認爲@"UIApplication";
第四個參數:代理類(delegate class),MainWindow.xib文件中遵循UIApplicationDelegate的類的類名,因 爲UIApplication定義了一個delegte變量,這個變量應該遵循UIApplicationDelegate,負責控制程序的運行,若是主 nib文件沒有這個類,你應該自定義一個這樣的類,並將第四個參數改成這個類的類名,不然這個程序不知道如何進行運做,由於前三個參數表明應用程序自己, 它除了把應用的事件循環啓動起來,並讀取info.plist裏的配置信息,不作其它任何事情。若是該參數爲nil,則程序假設程序的代理來自Main nib文件。
根據上面的分析,咱們來看如下iOS程序的聲明週期
對於UIApplicationMain函數中的第四個參數,咱們也能夠看出新舊版本的不一樣,咱們建議在原先的工程中使用新的版本,以提升程序的速度,共修改以下幾處
假如你的工程類都是以TC開頭。
1.import你的appdelegate類,並修改第四個參數以下:
UIApplicationMain(argc, argv, nil, NSStringFromClass([TCAppDelegate class]));
2.刪除MainWindow.xib文件
3.在工程的Info.plist文件中刪除下面一行
4.在TCAppDelegate.m文件中,修改
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions函數,以下:
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
// Override point for customization after application launch.
self.viewController = [[TCViewController alloc] initWithNibName:@"TCViewController" bundle:nil];
self.window.rootViewController = self.viewController;
[self.window makeKeyAndVisible];
return YES;
其中加粗部分爲新增部分。
通過上面的修改,咱們就能夠在程序load的過程當中,省去加載MainWindow.xib文件,提升程序的速度。
iPhone應用程序是由主函數main啓動,它負責調用UIApplicationMain函數,該函數的形式以下所示:
int UIApplicationMain (
int argc,
char *argv[],
NSString *principalClassName,
NSString *delegateClassName
);
那麼UIApplicationMain函數到底作了哪些事情呢?這個函數主要負責三件 事情:
1)從給定的類名初始化應用程序對象,也就是初始化UIApplication或者子類對象的一個實例,若是你在這裏給定的是nil,那麼 系統會默認UIApplication類,也就主要是這個類來控制以及協調應用程序的運行。在後續的工做中,你能夠用靜態方法 sharedApplication 來獲取應用程序的句柄。
2)從給定的應用程序委託類,初始化一個應用程序委託。並把該委託設置爲應用程序的委託,這裏就有若是傳入參數爲nil,會調用函數訪問 Info.plist文件來尋找主nib文件,獲取應用程序委託。
3)啓動主事件循環,並開始接收事件。
上面是UIApplicationMain函數的工做,接下來一個問題是應用程序視圖的顯示、消息的控制怎麼辦?下面就是UIApplication(或 者子類)對象的職責,這個對象主要作下面幾件事:
1)負責處理到來的用戶事件,並分發事件消息到應該處理該消息的目標對象(sender, action)。
2)管理以及控制視圖,包括呈現、控制行爲、當前顯示視圖等。
3)該對象有一個應用程序委託對象,當一些生命週期內重要事件(能夠包括系統事件或者生命週期控制事件)發生時,應用程序通知該對象。例如,應用程序啓 動、內存不夠了或者應用程序結束等,讓這些事件發生時,應用程序委託去響應。
通 過上面的分析,能夠知道UIApplication對開發者來講,是一個黑箱,它也能夠是。由於全部的操做,均可以由它的委託來幫咱們完成,它只須要在 後面維護一些不可更改的東西,如事件消息分發和傳遞、給委託發送事件處理請求等等,如,應用程序加載處理完畢,它會發送消息給委託,而後委託能夠在 applicationDidFinishLanching委託函數中去實現開發者想要的動做。利用XCODE在建立應用程序時,會默認實現一個應用程序 委託類。而對於加載的視圖,則有視圖相關的委託類來處理視圖加載過程的生命事件。下面說明委託主要能夠辦哪些事情:
控制應用程序的行爲
- (void)applicationDidFinishLaunching:(UIApplication *)application
應用程序啓動完畢。
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
當因爲其它方法打開應用程序(如URL指定或者鏈接),通知委託啓動完畢
- (void)applicationWillTerminate:(UIApplication *)application
通知委託,應用程序將在關閉 退出,請作一些清理工做。
- (void)applicationDidReceiveMemoryWarning:(UIApplication *)application
通知委託,應用程序收到了爲來自系統的內存不足警告。-(void)applicationSignificantTimeChange:(UIApplication *)application
通知委託系統時間發生改變(主要是指時間屬性,而不是具體的時間值)
打開URL
- (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url
打開指定的URL
控制狀態欄方位變化
– application:willChangeStatusBarOrientation:duration:
設備方向將要發生改變
– application:didChangeStatusBarOrientation:
活動狀態改變
- (void)applicationWillResignActive:(UIApplication *)application
通知委託應用程序將進入非活動狀態,在此期間,應用程序不接收消息或事件。-(void)applicationDidBecomeActive:(UIApplication *)application
通知委託應用程序進入活動狀態,請恢復數據
1.設置icon上的數字圖標
//設置主界面icon上的數字圖標,在2.0中引進, 缺省爲0
[UIApplicationsharedApplication].applicationIconBadgeNumber = 4;
2.設置搖動手勢的時候,是否支持redo,undo操做
//搖動手勢,是否支持redo undo操做。
//3.0之後引進,缺省YES
[UIApplicationsharedApplication].applicationSupportsShakeToEdit =YES;
3.判斷程序運行狀態
//判斷程序運行狀態,在2.0之後引入
if([UIApplicationsharedApplication].applicationState ==UIApplicationStateInactive){
NSLog(@"程序在運行狀態");
}
4.阻止屏幕變暗進入休眠狀態
//阻止屏幕變暗,慎重使用,缺省爲no 2.0
[UIApplicationsharedApplication].idleTimerDisabled =YES;
慎重使用本功能,由於很是耗電。
5.顯示聯網狀態
//顯示聯網標記 2.0
[UIApplicationsharedApplication].networkActivityIndicatorVisible =YES;
6.在map上顯示一個地址
NSString* addressText =@"1 Infinite Loop, Cupertino, CA 95014";
// URL encode the spaces
addressText = [addressTextstringByAddingPercentEscapesUsingEncoding:NSASCIIStringEncoding];
NSString* urlText = [NSStringstringWithFormat:@"http://maps.google.com/maps?q=%@", addressText];
[[UIApplicationsharedApplication]openURL:[NSURLURLWithString:urlText]];
7.發送電子郵件
NSString *recipients =@"mailto:first@example.com?cc=second@example.com,third@example.com&subject=Hello from California!";
NSString *body =@"&body=It is raining in sunny California!";
NSString *email = [NSStringstringWithFormat:@"%@%@", recipients, body];
email = [emailstringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
[[UIApplicationsharedApplication]openURL:[NSURLURLWithString:email]];
8.打電話到一個號碼
// Call Google 411
[[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"tel://8004664411"]];
9.發送短信
// Text to Google SMS
[[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"sms://466453"]];
10.打開一個網址
// Lanuch any iPhone developers fav site
[[UIApplicationsharedApplication]openURL:[NSURLURLWithString:@"http://itunesconnect.apple.com"]];
能夠看到UIApplication的頭文件實現 @interface UIApplication :UIResponder <UIActionSheetDelegate>{ @package id<UIApplicationDelegate> _delegate ; //這就是應用程序委託。 NSTimer ....... } 所以,在UIApplication中處理的系統事件時,只需轉到_delegate這個類去處理, 這個類對象就是應用程序委託對象。咱們能夠從應用程序的單例類對象中獲得應用程序委託的對象 UIApplicationDelegate* myDelegate = [[UIApplication sharedApplication] delegate]; UIApplication 接收到全部的系統事件和生命週期事件時,都會把事件傳遞給UIApplicationDelegate進行處理,對於用戶輸入 事件,則傳遞給相應的目標對象去處理。好比咱們在應用程序被來電等消息後,能夠調用應用程序委託類的 applicationWillResignActive()方法,這個方法在用戶鎖住屏幕時,也會調用,與之相適應的是應用程序從新被用戶打開時的委託 方法。另外經常使用的就是內存不足的系統警告,此時會調用應用程序委託類的applicationDidReceiveMemoryWarning()方法, 而後咱們就能夠試着釋放一些內存了。 上面就是應用程序生命週期(啓動,停止,恢復,退出等過程)的應用程序處理UIApplication sharedApplication