一占遥、簡介
分為本地推送和遠(yuǎn)程推送2種俯抖。可以在應(yīng)用沒有打開甚至手機(jī)鎖屏情況下給用戶以提示瓦胎。它們都需要注冊芬萍,注冊后系統(tǒng)會(huì)彈出提示框(如下圖)提示用戶是否同意,如果同意則正常使用搔啊;如果用戶不同意則下次打開程序也不會(huì)彈出該提示框柬祠,需要用戶到設(shè)置里面設(shè)置。一共有三種提示類型:
- UIUserNotificationTypeBadge:應(yīng)用圖標(biāo)右上角的信息提示
- UIUserNotificationTypeSound:播放提示音
- UIUserNotificationTypeAlert:提示框
二负芋、本地推送
1 注冊與處理
代碼如下:
/// 一般在在啟動(dòng)時(shí)注冊通知漫蛔,程序被殺死,點(diǎn)擊通知后調(diào)用此程序
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0) { // iOS8
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeAlert | UIUserNotificationTypeSound categories:nil];
[application registerUserNotificationSettings:setting];
}
if (launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]) {
// 這里添加處理代碼
}
return YES;
}
/// 程序沒有被殺死(處于前臺或后臺)旧蛾,點(diǎn)擊通知后會(huì)調(diào)用此程序
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification {
// 這里添加處理代碼
}
可以看到莽龟,處理代碼有兩個(gè)方法,一個(gè)是
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification;
另一個(gè)是
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
如果程序沒有被殺死锨天,即處于前臺或者后臺毯盈,那么調(diào)用前者;如果程序被殺死病袄,則調(diào)用后者搂赋。
2 發(fā)送通知
代碼如下
- (IBAction)addLocalNotification {
// 1.創(chuàng)建一個(gè)本地通知
UILocalNotification *localNote = [[UILocalNotification alloc] init];
// 1.1.設(shè)置通知發(fā)出的時(shí)間
localNote.fireDate = [NSDate dateWithTimeIntervalSinceNow:5];
// 1.2.設(shè)置通知內(nèi)容
localNote.alertBody = @"這是一個(gè)推送這是一個(gè)推送";
// 1.3.設(shè)置鎖屏?xí)r,字體下方顯示的一個(gè)文字
localNote.alertAction = @"趕緊!!!!!";
localNote.hasAction = YES;
// 1.4.設(shè)置啟動(dòng)圖片(通過通知打開的)
localNote.alertLaunchImage = @"../Documents/IMG_0024.jpg";
// 1.5.設(shè)置通過到來的聲音
localNote.soundName = UILocalNotificationDefaultSoundName;
// 1.6.設(shè)置應(yīng)用圖標(biāo)左上角顯示的數(shù)字
localNote.applicationIconBadgeNumber = 999;
// 1.7.設(shè)置一些額外的信息
localNote.userInfo = @{@"qq" : @"704711253", @"msg" : @"success"};
// 2.執(zhí)行通知
[[UIApplication sharedApplication] scheduleLocalNotification:localNote];
}
效果如下:
3 取消通知
// 取消所有本地通知
[application cancelAllLocalNotifications];
三、遠(yuǎn)程推送
與Android上我們自己實(shí)現(xiàn)的推送服務(wù)不一樣陪拘,Apple對設(shè)備的控制非常嚴(yán)格厂镇,消息推送的流程必須要經(jīng)過APNs(Apple Push Notification service).
一般情況下如果一個(gè)程序退到后臺就不能運(yùn)行代碼(Audio、VoIP等等可以在后臺運(yùn)行)左刽,或者程序退出后,那么它就和對應(yīng)應(yīng)用的后臺服務(wù)器斷開了鏈接酌媒,就收不到服務(wù)器發(fā)送的信息了欠痴,但是每臺設(shè)備只要聯(lián)網(wǎng)就會(huì)和蘋果的APNs服務(wù)器建立一個(gè)長連接(persistent IP connection),這樣只要通過蘋果的APNs服務(wù)器秒咨,我們自己的服務(wù)器就可以間接的和設(shè)備保持連接了喇辽,示意圖如下:
使用步驟:
1 Xcode設(shè)置
勾選Backgroud Modes -> Remote notifications,主要是iOS7之后雨席,蘋果支持后臺運(yùn)行菩咨,如果這里打開后,當(dāng)接收到遠(yuǎn)程推送后,程序在后臺也可以做一些處理抽米,如下圖所示:
2 遠(yuǎn)程推送的注冊與本地推送不同特占,iOS8.0前后也不同,代碼見下面云茸。
另外是目,在第一次使用推送時(shí),可能會(huì)有這樣的疑問:didFinishLaunchingWithOptions
會(huì)在每次打開程序時(shí)被調(diào)用标捺,那是不是每次都會(huì)調(diào)用注冊函數(shù)懊纳,每次都會(huì)彈窗詢問用戶"是否允許推送通知"?其實(shí)這個(gè)窗口只會(huì)在第一次打開程序時(shí)彈出一次亡容,無論用戶允許或不允許蘋果會(huì)記住用戶的選擇嗤疯,注冊函數(shù)調(diào)用多次對用戶也沒什么影響
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// iOS8之后和之前應(yīng)區(qū)別對待
if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0) {
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeAlert | UIUserNotificationTypeSound categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:settings];
} else {
[application registerForRemoteNotificationTypes:UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIUserNotificationTypeSound];
}
return YES;
}
/// 這個(gè)函數(shù)存在的意義在于:當(dāng)用戶在設(shè)置中關(guān)閉了通知時(shí),程序啟動(dòng)時(shí)會(huì)調(diào)用此函數(shù)闺兢,我們可以獲取用戶的設(shè)置
- (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings {
[application registerForRemoteNotifications];
}
3 如果注冊失敗身弊,比如沒有證書等等,會(huì)調(diào)用:
/// 注冊失敗調(diào)用
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
NSLog(@"遠(yuǎn)程通知注冊失斄星谩:%@",error);
}
4 獲取deviceToken
如果用戶同意阱佛,蘋果會(huì)根據(jù)應(yīng)用的 bundleID 和 手機(jī)UDID 生成 deviceToken,然后調(diào)用 application 的 didregister 方法返回 devicetoken,程序應(yīng)該把 devicetoken 發(fā)給應(yīng)用的服務(wù)器,服務(wù)器有義務(wù)將其存儲(如果允許多點(diǎn)登錄,可能存多個(gè) devicetoken)。deviceToken也是會(huì)變的: ”If the user restores backup data to a new device or computer, or reinstalls the operating system, the device token changes“戴而,因此應(yīng)每次都發(fā)給服務(wù)器(provider)
/// 用戶同意后凑术,會(huì)調(diào)用此程序,獲取系統(tǒng)的deviceToken所意,應(yīng)把deviceToken傳給服務(wù)器保存淮逊,此函數(shù)會(huì)在程序每次啟動(dòng)時(shí)調(diào)用(前提是用戶允許通知)
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
NSLog(@"deviceToken = %@",deviceToken);
}
5 用戶點(diǎn)擊了通知
默認(rèn)會(huì)打開程序。處理代碼有三個(gè)函數(shù)扶踊,分iOS7之前之后和程序是否處于后臺
- 5.1 iOS7及其之之后
此函數(shù)無論是程序被殺死還是處于后臺泄鹏,只要用戶點(diǎn)擊了通知,都會(huì)被調(diào)用秧耗,因此如果是iOS7备籽,則不必在didFinishLaunchingWithOptions中做處理,只在下面函數(shù)做處理即可分井,此時(shí)應(yīng)避免在didFinishLaunchingWithOptions函數(shù)中也做重復(fù)處理车猬。
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
// userInfo
}
注:當(dāng)在第一步打開后臺運(yùn)行后,用戶不點(diǎn)擊通知尺锚,也可以執(zhí)行:
- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo fetchCompletionHandler:(void(^)(UIBackgroundFetchResult))completionHandler
- 5.2 iOS7之前
當(dāng)用戶點(diǎn)擊通知后珠闰,如果程序被殺死則會(huì)調(diào)用下面第一個(gè)函數(shù),如果程序處于后臺會(huì)調(diào)用下面第二個(gè)函數(shù)瘫辩,因此下面兩個(gè)函數(shù)應(yīng)搭配使用
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 獲取遠(yuǎn)程推送消息
NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey];
if (userInfo) {
// 有推送的消息伏嗜,處理推送的消息
}
return YES;
}
/// iOS3之后才有坛悉,只有在程序處于后臺時(shí),用戶點(diǎn)擊了通知后才會(huì)被調(diào)用承绸,應(yīng)搭配didFinishLaunchingWithOptions使用
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
// userInfo
}
在實(shí)際編程時(shí)裸影,如果想兼容iOS7以前,三個(gè)函數(shù)可同時(shí)使用八酒,都列出來空民,系統(tǒng)會(huì)自動(dòng)選擇合適的調(diào)用。
6 總結(jié)下函數(shù)的調(diào)用:
首次安裝后啟動(dòng):
- didRegisterForRemoteNotificationsWithDeviceToken 被調(diào)用
- 系統(tǒng)詢問用戶是否同意接收 Notifications
- 不管用戶選擇同意或拒絕羞迷,didRegisterUserNotificationSettings 被調(diào)用
應(yīng)用非首次啟動(dòng)時(shí):
如果 notifications 處于拒絕狀態(tài):didRegisterUserNotificationSettings 被調(diào)用
-
如果 notifications 處于允許狀態(tài)
- didRegisterForRemoteNotificationsWithDeviceToken 被調(diào)用
- didRegisterUserNotificationSettings 被調(diào)用
-
應(yīng)用運(yùn)行過程中用戶修改 notifications 設(shè)置:
- 從拒絕變?yōu)樵试S:didRegisterForRemoteNotificationsWithDeviceToken 被調(diào)用
- 從允許變?yōu)榫芙^:什么也不發(fā)生
7 服務(wù)端推送的格式
{
"aps" : { // 必須有
"alert" : "string",
"body" : "string",
"badge" : number,
"sound" : "string"
},
"NotiId" : 20150821, // 自定義key值
}
8 推送的大小限制
遠(yuǎn)程通知負(fù)載的大小根據(jù)服務(wù)器使用的API不同而不同界轩。當(dāng)使用HTTP/2 provider API時(shí),負(fù)載最大為4kB衔瓮;當(dāng)使用legacy binary interface時(shí)浊猾,負(fù)載最大為2kB。當(dāng)負(fù)載大小超過規(guī)定的負(fù)載大小時(shí)热鞍,APNs會(huì)拒絕發(fā)送此通知葫慎。