IOS 應用程式啟動載入過程(從點擊表徵圖到介面顯示),ios載入
今天幫同事解決問題的時候發現,造成程式BUG的原因是同事對程式的啟動和試圖的載入過程不熟悉。所以當局部代碼沒有問題,但是程式一運行卻總不是我們想要結果的時候,我們應該想想是不是因為我們忽略了試圖載入過程的原因。下面我們用一個例子來簡單介紹一下啟動過程中常見的幾個方法。首先我們通過XIB 建立了一個試圖控制器(名字叫Empty ,不要問我為為啥它叫這個名字,因為我太懶了,滑鼠點快了就沒去該),又建立了一個UIView的子類(MyView),並且將Empty的對象 設定為視窗的跟控制器,MyView 的一個對象設定為Empty的主視圖。OK,瞭解了以上的內容,我們就可以寫代碼了。
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { NSLog(@"%s",__func__); //載入控制器 Empty* test=[[Empty alloc] initWithNibName:@"Empty" bundle:nil]; //建立視窗 UIWindow *window=[[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds]; self.window=window; //將建立的控制器設定為根控制器 self.window.rootViewController=test; //啟用並顯示視窗 [self.window makeKeyAndVisible]; return YES;}
由於我們的應用程式並不是直接通過故事板建立的,所以我們要自己設定跟控制器。在應用程式調用didFinishLaunchingWithOptions:之前,會先查看是否能通過故事板來建立視圖控制器,如果找不到的話則需要我們手動建立window和控制器。由於我們想知道各個方法的執行過程,所以我們要在各個方法中加上方法的列印資訊。Empty視圖控制器代碼如下:
#import "MyView.h"#import "Empty.h"@interface Empty ()@end@implementation Empty//再程式一啟動的時候就調用各個類的load方法 正如官方說說的 無論這個類是否會載入到runtime中,它都會被調用。//Invoked whenever a class or category is added to the Objective-C runtime;+(void)load{ NSLog(@"%s",__func__);}//這個方法只會調用一次,再init之前。如果一個類建立了10個對象,那麼init會執行10次,但是該方法只會執行一次。+(void)initialize{ NSLog(@"%s",__func__);}//這個就不說了 太熟悉了-(instancetype)init{ NSLog(@"%s",__func__); return [super init];}//不論是-(instancetype)initWithCoder:(NSCoder *)aDecoder{ NSLog(@"%s",__func__); return [super initWithCoder:aDecoder];}//當從nib 中載入的時候 會調用這個方法-(instancetype)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil{ NSLog(@"%s",__func__); if (self=[super initWithNibName:nibNameOrNil bundle:nibBundleOrNil]) { } return self;}//初始化完成後就開始載入視圖,注意一點,控制器建立完成了並不代表它的視圖已經載入好了,視圖是懶載入的,使用時應注意。-(void)loadView{ NSLog(@"loadview"); [super loadView]; UIView *view=[[MyView alloc] init]; view.backgroundColor=[UIColor yellowColor]; self.view=view; }// 當觸發約束時調用-(void)updateViewConstraints{ NSLog(@"%s",__func__); [super updateViewConstraints]; }//當視圖載入完成後調用,如果視圖不是被銷毀後再重新顯示的話,那麼它只會執行一次- (void)viewDidLoad{ NSLog(@"%s",__func__); [super viewDidLoad]; }//視圖即將已經顯示-(void)viewDidAppear:(BOOL)animated{ [super viewDidAppear:animated]; NSLog(@"%s",__func__);}//視圖即將顯示-(void)viewWillAppear:(BOOL)animated{ [super viewWillAppear:animated]; NSLog(@"%s",__func__);}
以上是控制器類,該說的再代碼裡都說了。下面是MyView中的代碼。
#import "MyView.h"@implementation MyView//繪圖- (void)drawRect:(CGRect)rect{ NSLog(@"%s",__func__);}+(void)load{ NSLog(@"%s",__func__);}// 為子視圖設定布局 一般在這裡進行各個子控制項的frame-(void)layoutSubviews{ [super layoutSubviews]; NSLog(@"%s",__func__);}@end
看完代碼,我們再來看看列印資訊。
2015-04-21 00:26:00.123 Runloop[19756:1191521] +[Empty load]2015-04-21 00:26:00.124 Runloop[19756:1191521] +[MyView load]2015-04-21 00:26:00.309 Runloop[19756:1191521] -[AppDelegate application:didFinishLaunchingWithOptions:]2015-04-21 00:26:00.309 Runloop[19756:1191521] +[Empty initialize]2015-04-21 00:26:00.309 Runloop[19756:1191521] -[Empty initWithNibName:bundle:]2015-04-21 00:26:00.310 Runloop[19756:1191521] loadview2015-04-21 00:26:00.314 Runloop[19756:1191521] -[Empty viewDidLoad]2015-04-21 00:26:00.314 Runloop[19756:1191521] -[Empty viewWillAppear:]2015-04-21 00:26:00.315 Runloop[19756:1191521] -[MyView layoutSubviews]2015-04-21 00:26:00.315 Runloop[19756:1191521] -[MyView layoutSubviews]2015-04-21 00:26:00.316 Runloop[19756:1191521] -[MyView drawRect:]2015-04-21 00:26:00.350 Runloop[19756:1191521] -[Empty viewDidAppear:]
請忽略項目名,因為今天本來想些runloop的,唉~世事難料啊~~~