iOS開發UI篇—程序啓動原理和UIApplication

1、UIApplication
1.簡單介紹

(1)UIApplication對象是應用程序的象徵,一個UIApplication對象就表明一個應用程序。ios

(2)每個應用都有本身的UIApplication對象,並且是單例的,若是試圖在程序中新建一個UIApplication對象,那麼將報錯提示。網絡

(3)經過[UIApplication sharedApplication]能夠得到這個單例對象app

(4)一個iOS程序啓動後建立的第一個對象就是UIApplication對象,且只有一個(經過代碼獲取兩個UIApplication對象,打印地址能夠看出地址是相同的)。ide

(5)利用UIApplication對象,能進行一些應用級別的操做函數

2.應用級別的操做示例:oop

1)設置應用程序圖標右上角的紅色提醒數字(如QQ消息的時候,圖標上面會顯示1,2,3條新信息等。)動畫

@property(nonatomic) NSInteger applicationIconBadgeNumber;atom

代碼實現和效果:url

 1 - (void)viewDidLoad
 2 {
 3     [super viewDidLoad];
 4     //建立並添加一個按鈕
 5     UIButton *btn=[[UIButton alloc]initWithFrame:CGRectMake(100, 100, 60, 30)];
 6     [btn setTitle:@"按鈕" forState:UIControlStateNormal];
 7     [btn setBackgroundColor:[UIColor brownColor]];
 8     [btn addTarget:self action:@selector(onClick) forControlEvents:UIControlEventTouchUpInside];
 9     [self.view addSubview:btn];
10 }
11 -(void)onClick
12 {
13     NSLog(@"按鈕點擊事件");
14     //錯誤,只能有一個惟一的UIApplication對象,不能再進行建立
15 //    UIApplication *app=[[UIApplication alloc]init];
16     
17     //經過sharedApplication獲取該程序的UIApplication對象
18     UIApplication *app=[UIApplication sharedApplication];
19     app.applicationIconBadgeNumber=123;
20 }

2)設置聯網指示器的可見性spa

@property(nonatomic,getter=isNetworkActivityIndicatorVisible) BOOL networkActivityIndicatorVisible;

代碼和效果:

   // 設置指示器的聯網動畫
    app.networkActivityIndicatorVisible=YES;

3)管理狀態欄

從iOS7開始,系統提供了2種管理狀態欄的方式

a.經過UIViewController管理(每個UIViewController均可以擁有本身不一樣的狀態欄).

在iOS7中,默認狀況下,狀態欄都是由UIViewController管理的,UIViewController實現下列方法就能夠輕鬆管理狀態欄的可見性和樣式

狀態欄的樣式     - (UIStatusBarStyle)preferredStatusBarStyle; 

狀態欄的可見性  -(BOOL)prefersStatusBarHidden;

 1 #pragma mark-設置狀態欄的樣式
 2 -(UIStatusBarStyle)preferredStatusBarStyle
 3 {
 4     //設置爲白色
 5     //return UIStatusBarStyleLightContent;
 6     //默認爲黑色
 7      return UIStatusBarStyleDefault;
 8 }
 9 #pragma mark-設置狀態欄是否隱藏(否)
10 -(BOOL)prefersStatusBarHidden
11 {
12     return NO;
13 }

b.經過UIApplication管理(一個應用程序的狀態欄都由它統一管理)

若是想利用UIApplication來管理狀態欄,首先得修改Info.plist的設置

 

代碼:

 1 //經過sharedApplication獲取該程序的UIApplication對象
 2     UIApplication *app=[UIApplication sharedApplication];
 3     app.applicationIconBadgeNumber=123;
 4     
 5     //設置指示器的聯網動畫
 6     app.networkActivityIndicatorVisible=YES;
 7     //設置狀態欄的樣式
 8     //app.statusBarStyle=UIStatusBarStyleDefault;//默認(黑色)
 9     //設置爲白色+動畫效果
10       [app setStatusBarStyle:UIStatusBarStyleLightContent animated:YES];
11     //設置狀態欄是否隱藏
12     app.statusBarHidden=YES;
13       //設置狀態欄是否隱藏+動畫效果
14     [app setStatusBarHidden:YES withAnimation:UIStatusBarAnimationFade];

c.補充

既然兩種均可以對狀態欄進行管理,那麼何時該用什麼呢?
若是狀態欄的樣式只設置一次,那就用UIApplication來進行管理;
若是狀態欄是否隱藏,樣式不同那就用控制器進行管理。
UIApplication來進行管理有額外的好處,能夠提供動畫效果。

 4)openURL:方法

UIApplication有個功能十分強大的openURL:方法

- (BOOL)openURL:(NSURL*)url;

openURL:方法的部分功能有

打電話  UIApplication *app = [UIApplicationsharedApplication]; [app openURL:[NSURLURLWithString:@"tel://10086"]];

發短信  [app openURL:[NSURLURLWithString:@"sms://10086"]];

發郵件  [app openURL:[NSURLURLWithString:@"mailto://12345@qq.com"]];

打開一個網頁資源 [app openURL:[NSURLURLWithString:@"http://ios.itcast.cn"]];

打開其餘app程序   openURL方法,能夠打開其餘APP。

  URL補充:
URL:統一資源定位符,用來惟一的表示一個資源。
URL格式:協議頭:// 主機地址/資源路徑
網絡資源:http\ftp等   表示百度上一張圖片的地址   http://www.baidu.com/images/20140603/abc.png
本地資源:file:///users/apple/desktop/abc.png(主機地址省略)
2、UIApplication Delegate

1.簡單說明

全部的移動操做系統都有個致命的缺點:app很容易受到打擾。好比一個來電或者鎖屏會致使app進入後臺甚至被終止。

還有不少其它相似的狀況會致使app受到干擾,在app受到干擾時,會產生一些系統事件,這時UIApplication會通知它的delegate對象,讓delegate代理來處理這些系統事件。

做用:當被打斷的時候,通知代理進入到後臺。

每次新建完項目,都有個帶有「AppDelegate」字眼的類,它就是UIApplication的代理,NJAppDelegate默認已經遵照了UIApplicationDelegate協議,已是UIApplication的代理。

2.代理方法
 1 #import "YYAppDelegate.h"
 2 
 3 @implementation YYAppDelegate
 4 
 5 // 當應用程序啓動完畢的時候就會調用(系統自動調用)
 6 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
 7 {
 8     NSLog(@"didFinishLaunchingWithOptions");
 9     return YES;
10 }
11 
12 // 即將失去活動狀態的時候調用(失去焦點, 不可交互)
13 - (void)applicationWillResignActive:(UIApplication *)application
14 {
15     NSLog(@"ResignActive");
16 }
17 
18 // 從新獲取焦點(可以和用戶交互)
19 - (void)applicationDidBecomeActive:(UIApplication *)application
20 {
21     NSLog(@"BecomeActive");
22 }
23 
24 // 應用程序進入後臺的時候調用
25 // 通常在該方法中保存應用程序的數據, 以及狀態
26 - (void)applicationDidEnterBackground:(UIApplication *)application
27 {
28     NSLog(@"Background");
29 }
30 
31 // 應用程序即將進入前臺的時候調用
32 // 通常在該方法中恢復應用程序的數據,以及狀態
33 - (void)applicationWillEnterForeground:(UIApplication *)application
34 {
35     NSLog(@"Foreground");
36 }
37 
38 // 應用程序即將被銷燬的時候會調用該方法
39 // 注意:若是應用程序處於掛起狀態的時候沒法調用該方法
40 - (void)applicationWillTerminate:(UIApplication *)application
41 {
42 }
43 
44 // 應用程序接收到內存警告的時候就會調用
45 // 通常在該方法中釋放掉不須要的內存
46 - (void)applicationDidReceiveMemoryWarning:(UIApplication *)application
47 {
48     NSLog(@"MemoryWarning");
49 }
50 @end
應用程序通常有五個狀態:官方文檔app.states
 
3、程序啓動原理

UIApplicationMain

main函數中執行了一個UIApplicationMain這個函數

intUIApplicationMain(int argc, char *argv[], NSString *principalClassName, NSString *delegateClassName);

argc、argv:直接傳遞給UIApplicationMain進行相關處理便可 

principalClassName:指定應用程序類名(app的象徵),該類必須是UIApplication(或子類)。若是爲nil,則用UIApplication類做爲默認值

 delegateClassName:指定應用程序的代理類,該類必須遵照UIApplicationDelegate協議

UIApplicationMain函數會根據principalClassName建立UIApplication對象,根據delegateClassName建立一個delegate對象,並將該delegate對象賦值給UIApplication對象中的delegate屬性

接着會創建應用程序的Main Runloop(事件循環),進行事件的處理(首先會在程序完畢後調用delegate對象application:didFinishLaunchingWithOptions:方法)

1 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;

 

程序正常退出時UIApplicationMain函數才返回

 1 #import <UIKit/UIKit.h>
 2 
 3 #import "YYAppDelegate.h"
 4 
 5 int main(int argc, char * argv[])
 6 {
 7     @autoreleasepool {
 8         // return UIApplicationMain(argc, argv, nil, NSStringFromClass([YYAppDelegate class]));
 9         // return UIApplicationMain(argc, argv, @"UIApplication", NSStringFromClass([YYAppDelegate class]));
10         /*
11          argc: 系統或者用戶傳入的參數個數
12          argv: 系統或者用戶傳入的實際參數
13          1.根據傳入的第三個參數建立UIApplication對象
14          2.根據傳入的第四個產生建立UIApplication對象的代理
15          3.設置剛剛建立出來的代理對象爲UIApplication的代理
16          4.開啓一個事件循環
17          */
18          return UIApplicationMain(argc, argv, @"UIApplication", @"YYAppDelegate");
19     }
20 }

系統入口的代碼和參數說明:

argc:系統或者用戶傳入的參數
argv:系統或用戶傳入的實際參數 
1.根據傳入的第三個參數,建立UIApplication對象
2.根據傳入的第四個產生建立UIApplication對象的代理
3.設置剛剛建立出來的代理對象爲UIApplication的代理
4.開啓一個事件循環(能夠理解爲裏面是一個死循環)這個時間循環是一個隊列(先進先出)先添加進去的先處理
 
ios程序啓動原理

4、程序啓動的完整過程

1.main函數

2.UIApplicationMain

* 建立UIApplication對象

* 建立UIApplication的delegate對象

 

3.delegate對象開始處理(監聽)系統事件(沒有storyboard)

* 程序啓動完畢的時候, 就會調用代理的application:didFinishLaunchingWithOptions:方法

* 在application:didFinishLaunchingWithOptions:中建立UIWindow

* 建立和設置UIWindow的rootViewController

* 顯示窗口

 

3.根據Info.plist得到最主要storyboard的文件名,加載最主要的storyboard(有storyboard)

* 建立UIWindow

* 建立和設置UIWindow的rootViewController

* 顯示窗口

相關文章
相關標籤/搜索