程序启动
// UIApplicationMain主要负责了:从给定的类名初始化应用程序对象、从给定的应用程序委托类,初始化一个应用程序委托、启动主事件循环,并开始接收事件
// 第三个参数 principalClassName -- UIApplication 或 UIApplication 子类,nil 默认为 UIApplication
// 第四个参数 delegateClassName -- AppDelagate 类名
int main(int argc, char * argv[]) {
@autoreleasepool {
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
}
可以认为,一个 UIApplication 就是一个 APP 的代表
UIApplication
[UIApplication sharedApplication]
可以获取 UIApplication 的单例
UIApplicationState 为枚举类型,表示 UIApplication 的状态,有:
- UIApplicationStateActive
- UIApplicationStateInactive
- UIApplicationStateBackground
UIApplication 可以做的事情有
// icon 上的数字提醒
.applicationIconBadgeNumber
// 打开系统或第三方的 APP
- openURL:
- canOpenURL:
// 控制屏幕常亮
.idleTimerDisabled
<UIApplicationDelegate>
UIApplication 有属性 .delegate ,AppDelegate 实现了 <UIApplicationDelegate> 这个协议
AppDelegate 做的事情有:
- launch -- 负责app的启动
- window/rootVc -- 管理一个app的window和rootVc
- application state -- 托管了整个 app 的状态
- push -- 管理 app 所有的 push 消息推送
- ...
launch
// launchOptions — 表示app启动的选项,代表它从不同来源触发的app启动,有 RemoteNotificationKey( 通过远程push启动 )、URLKey( 从 openURL 启动 )、SourceApplicationKey( 从哪个app调用openURL启动 )、ShortcutItemKey( 快捷方式启动APP )...
- application:didFinishLaunchingWithOptions:
openUrl 打开第三方 app
从 iOS9 开始无法直接使用 canOpenUrl 判断用户是否安装了某个 app,需要对这些 app 的 url scheme 加入白名单,在 info.plist 里的 LSApplicationQueriesSchemes 项增加白名单。白名单有50个数量上限
在工程的 info - URL types - URL Schemes 里指定自己的APP的 URL Schemes
openUrl 打开第三方 app:
NSURL *url = [NSURL URLWithString:@"someURLScheme://"];
if ([[UIApplication sharedApplication] canOpenURL:url]) {
[[UIApplication sharedApplication] openURL:url];
}
APP launch by URL Schemes
<UIApplicationDelegate> 协议定义了两个方法,当 APP 由 URL Scheme 打开时调用这两个方法
// 如果 APP 未启动,则先执行 didFinishLaunchingWithOptions ,如果 APP 已经启动,则直接执行方法
// after iOS9
- application:openURL:options:
// before iOS9
- application:openURL:sourceApplication:annotation:
在调试时,如果希望用 URL Schemes 启动,又能正常进行调试,可以选择 URL Schemes 启动调试:Product - Scheme - Edit Scheme - Run - info - Launch - Wait for executable to be launched
window/rootVc
- key window
用来接收键盘和非触摸类的消息。key window 指当前活跃的 window,即在某一个时间点上,只有一个 window
可通过[[UIApplication sharedApplication] keyWindow]
来访问,((AppDelegate *)[UIApplication sharedApplication].delegate).window
也是 key window - window level
UIWindowLevelNormal 最低
UIWindowLevelAlert 最高
UIWindowLevelStatusBar 中等 - 通常在 didFinishLaunchingWithOptions 中创建 window 并设置它的 rootVc
Application State
Application State Callback
在 AppDelegate 有关于 APP 生命周期的回调
// avtive <=> inactive
// app 从没有启动( not running )到启动( running )的过程
- applicationDidBecomeActive:
// 在 APP 启动时唤起控制中心
- applicationWillResignActive:
// foreground <=> background
// 按 home 键回到桌面
- applicationDidEnterBackground:
// 从桌面返回 APP
- applicationWillEnterForeground:
// APP 在后台放很久后被结束,用户主动将 app 放进后台并杀掉的话,并不会触发(如果没进入 background 就杀掉就会触发)
- applicationWillTerminate:APP 在后台中的运行
iOS 对于后台运行有严格的权限控制,除了明确可以在后台的任务,如播放音乐,其他不能长时间后台运行,但可以在进入后台后短时间执行如数据保存等,从 iOS7 开始只有3分钟
[UIApplication class]
- beginBackgroundTaskWithExpirationHandler:
- beginBackgroundTaskWithName:expirationHandler:
- AppDelegate 以外获取 APP 状态的变化 -- Notification
获取 NSNotificationCenter 实例:[NSNotificationCenter defaultCenter]
[NSNotificationCenter class]
- addObserver:selector:name:object:
- postNotificationName:object:userInfo:
// 如果 deployment target 从 9.0 开始,就不需要再关心 removeObserver
- removeObserver:
Push
受到Push后的处理
// App running
- application:handleActionWithIdentifier:forRemoteNotification:completionHandler:
// App not running
- application:didFinishLaunchingWithOptions:
3D Touch
- application:performActionForShortcutItem:completionHandler:
Universal Link
// 从第三方的应用打开,spotlight search 也会调用
- application:continueUserActivity:restorationHandler: