一、PCH 文件
PCH 是一個頭文件,能被項目中的所有源文件共享和訪問。
- PCH 文件的需求
一個宏或者頭文件等,很多文件都需要用到,怎么解決,搞個共用的頭文件,同時導入這個頭文件。 - 作用
a. 存放一些共用的宏
b. 存放一些共用的頭文件
c. 管理日志的輸入,自定義LOG - 為什么要管理日志輸出
因為日志輸出非常耗性能,一般發布的時候不需要日志輸出,只有調試的時候才需要。 - 注意
在PCH 中寫有關OC的方法,最好放在#ifdef OBJC中,xcode 在每個OC文件中都定義了這個宏,也就意味著只有OC文件才擁有這些宏,避免了項目文件中有C文件報錯。
二、程序啟動原理
- 程序啟動過程
a. 打開程序
b. 執行main 函數
c. 結束程序 - 執行Main 函數
a. UIApplicationMian 函數原型:UIKIT_EXTERN int UIApplicationMain(int argc, char *argv[], NSString * __nullable principalClassName, NSString * __nullable delegateClassName);
b. UIApplicationMian 的底層實現:
(1). 根據principalClassName 提供的類名,創建一個UIApplication 對象。- UIApplication 代表一個應用程序
- UIApplication 一般用來做一些應用級別的操作(app 的提醒框,聯網狀態,打電話,打開網頁,控制狀態)
(2). 根據delegateClassName 提供的類名,創建一個UIApplication 對象。 - 程序加載完畢時調用:application:didFinishLaunchingWithOptions:
- 程序獲取焦點時調用:applicationDidBecomeActive:
- 程序進入后臺時調用:applicationDidEnterBackgroup:
- 程序失去焦點時調用:applicationWillResignActive:
- 程序從后臺進入前臺時調用:applicationWillEnterForegroup:
- 內存警告,可能要終止程序時調用:applicationDidReceviceMemoryWarning:
- 程序即將退出時調用:applicationWillTerminate:
(3). 開啟一個主運行循環,它是保存程序一直運行,并處理事件。
(4). 加載info.plist 和啟動圖片,并且判斷info.plist 有沒有指定Main.storyboard,如果指定就去加載。
- application 隱藏狀態欄
(1). 設置Info.plist文件:添加健View controller-based status bar appearance,設置值為NO。
(2). 創建application。
(3). 調用隱藏狀態欄的方法。 - 補充:反射機制
反射機制的好處:如果類名用字符串表示,即使類名寫錯了,編輯器不會報錯;通過反射機制類名寫錯了,編譯器報錯。
NSString * class = NSStringFromClass([AppDelegate class]);
appDelegate * strClass = NSClassFromString(@"AppDelegate");
三、加載Main.storyboard
- 加載Main.storyboard 步驟
a. 創建窗口
b. 加載Main.storyboard,并且加載Main.storyboard指定的控制器
c. 把新的控制器作為窗口的根控制器,并讓窗口顯示出來 - 窗口Window
a. UIWindow 是一個特殊的UIView, 在一個app 中一般都會有一個UIWindow ,但不僅只有一個,例如:鍵盤也是一個窗口
b. app 程序啟動完成后,創建的第一個視圖控件就是UIWindow, 接著創建控制器的UIView,最后將控制器添加到Window上,于是控制器的view 就顯示在屏幕上了。
c. 一個app 之所以能顯示在屏幕上,完全是因為有UIWindow。
d. UIScreen :標識物理的屏幕,他連接著設備。
e. UIWindow :用于提供繪制支持,提供了一些繪制方法。
f. UIView :窗口上有很多view,是用于繪圖操作的,把畫好的view 添加到窗口上;屏幕上的東西都是繪制上去的,刷新一遍相當于重新繪制一遍。
g. 只有加載Main.storyborad 的時候才創建窗口(加載:系統自動加載)
h. 如果是自己用代碼加載Main.storyborad 需要自己創建窗口代碼。 - 補充
a. 如果把新建的控制器的view用addSubview: 方法直接添加到窗口上,不會有旋轉功能。
b. 設置窗口的根視圖控制器RootViewController,會自動把控制器的view添加到窗口。
c. 查看主窗口:application.keyWindow
d. 顯示窗口:self.window.hidden = NO
e. 查看程序的所有窗口:application.windows - addSubView 和rootViewController 的區別
a. 直接用addSubView,控制器會被釋放,控制器就不能處理事件。
b. 直接用addSubView,控制器的view不會旋轉。
c. 用rootViewController,控制器不會被釋放,而且控制器的view會自動旋轉。
d. 旋轉事件->UIApplication -> window -> rootViewController -> 旋轉控制器view - makeKeyAndVisible 方法底層做的事
a. 把窗口設置成主窗口,如:application.keyWindow = self.window;
b. 顯示窗口,如:self.window.hidden = NO;
c. 注意:雖然底層會做上面兩步,但不一定是上面的代碼。 - 窗口的層級
a. windowLevel: UIWindowLevelNormal < UIWindowLevelStatusBar < UIWindowLevelAlert
b. UIWindowLevelNormal :默認頂層層級
c. UIWindowLevelStatusBar :狀態欄、鍵盤
d. UIWindowLevelAlert:UIAlertView 、UIActionSheet
e. 把window 的層級設置成UIWindowLevelAlert ,就會顯示在最前面了。
f. 相同層級的窗口,想讓其中一個顯示,可以用那個窗口的層級加上一個數 - 代碼模仿main.storyboard 加載
self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
UIStoryboard * storyboard = [UIStoryboard storyboardWithName:@"main" bundle:nil];
UIViewController * vc = [storyboard instantiateViewControllerWithIdentifier:@"B"]
self.window.rootViewController = vc;
四、通過XIB 創建控制器的View
- 步驟
a. 創建一個控制器類
b. 創建一個xib,并指定所描述的控制器,一個xib 只能用來描述一個控制器,如果沒有指定,就不能拖線指定控制器的view
c. 拖線指定xib 中的哪個UIView 是控制器的view - 只有控制器的init 的方法,底層才會調用initWithNibName:bundle:方法
- UIView 的創建
a. 如果重寫loadView ,就根據自定義view 去創建view
b. 如果沒有重寫loadView,就去查看有沒有storyboard,有storyboard,就根據storyboard 描述的view去創建
c. 如果沒有storyboard 就去看是否 有對于的XIB,有xib 就根據xib 的描述的view創建
d. 如果沒有xib, 即nibName == nil 時,就查有沒有同名的xib,但是優先查看沒有controller 的xib, 如果查不到與xib完全同名的xib,如:xib的擁有者是viewController ,xib 的文件名就是view,就優先查view.xib,根據它的描述的view創建;如果沒有文件名的xib,就去查看名字為ViewController 的xib, 如果有就根據xib 里的描述的view 創建 - 控制器的loadView方法
a. loadView的作用:自定義控制器view, 只要重寫這個方法,說明要創建view,就不會自動創建view
b. loadView 說明時候調用:第一次使用view的時候調用,調用這個方法創建控制器的view
c. loadView 默認做法:如果storyboard 描述了控制器view,就去加載
d. 注意:1.只要重寫loadView方法,沒有調用系統默認的做法,即不寫[super loadView],就不會去加載storyboard或者xib來描述控制器的view 2.如果重寫loadView方法,并且指定了nibName,loadView默認的做法會去加載xib的view 3. 只要重寫loadView方法,沒有指定nibName,就不會自動去加載和控制器同名的xib 4. 在重寫loadView時,沒有給self.view創建view,就使用self.view,會造成死循環 5. 如果是根控制器的view,自定義view的時候可以不設置尺寸,系統會自動設置;不是跟控制器就不行;可以用CGRctZeco表示,如:self.view = [[UIView alloc] initWithFrame: CGRctZeco]; 6. 重寫loadView方法時,不要寫[super loadView];,因為重寫該方法的目的是自定義view,重寫了還要去加載storyboard里的view,等于多此一舉 - xib 和storyboard 的區別
storyboard 已經指定了控制器view,不需要我們管,xib 需要我們手動管理。 - 如何快速生成一個xib 描述的控制器view
1、定義新的控制器的時候,勾選xib,會自動搞一個xib描述控制器的view
2、會自動生成一個和控制器同名的xib,并且里面設置好了
五、控制器view
- view的生命周期:只要是view開頭的都是view的生命周期方法
loadView:第一次使用view的時候調用
viewDidLoad:控制器的view加載完成的時候調用
viewWillAppear:控制器的view即將顯示的時候調用
viewDidAppear:控制器的view完全顯示的時候調用
viewWillDisappear:控制器的view即將消失的時候調用
viewDidDisappear:控制器的view完全消失的時候調用
viewWillLayoutSubviews:控制器的view即將布局的時候調用
viewDidLayoutSubviews:控制器的view完全布局的時候調用
viewWillUnload:控制器的view即將銷毀
viewDidUnload:控制器的view完全銷毀 - 內存警告處理
a、處理過程
有內存警告 -> 調用didReceiveMemoryWarning方法 -> 判斷控制器的View存不
存在 ->
存在就判斷能不能被釋放(判斷是不是正在顯示在界面上) -> 能釋放就調用ViewWillUnload ->
完全釋放后就調用ViewDidUnload
b、注意
內存警告處理時,ViewWillUnload和ViewDidUnload不一定被調用,因為這是系統自動判斷的