一、簡介
分為本地推送和遠程推送2種。可以在應用沒有開啟甚至手機鎖屏情況下給使用者以提示。它們都需要註冊,註冊後系統會彈出提示框(如下圖)提示使用者是否同意,如果同意則正常使用;如果使用者不同意則下次開啟程式也不會彈出該提示框,需要使用者到設定裡面設定。一共有三種提示類型:
UIUserNotificationTypeBadge:應用表徵圖右上方的資訊提示
UIUserNotificationTypeSound:播放提示音
UIUserNotificationTypeAlert:提示框
二、本地推送
1 註冊與處理
代碼如下:
/// 一般在在啟動時註冊通知,程式被殺死,點擊通知後調用此程式- (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;}/// 程式沒有被殺死(處於前台或後台),點擊通知後會調用此程式- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification { // 這裡添加處理代碼}
可以看到,處理代碼有兩個方法,一個是
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification;
另一個是
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
如果程式沒有被殺死,即處於前台或者後台,那麼調用前者;如果程式被殺死,則調用後者。
2 發送通知
代碼如下
- (IBAction)addLocalNotification { // 1.建立一個本地通知 UILocalNotification *localNote = [[UILocalNotification alloc] init]; // 1.1.設定通知發出的時間 localNote.fireDate = [NSDate dateWithTimeIntervalSinceNow:5]; // 1.2.設定通知內容 localNote.alertBody = @"這是一個推送這是一個推送"; // 1.3.設定鎖屏時,字型下方顯示的一個文字 localNote.alertAction = @"趕緊!!!!!"; localNote.hasAction = YES; // 1.4.設定啟動圖片(通過通知開啟的) localNote.alertLaunchImage = @"../Documents/IMG_0024.jpg"; // 1.5.設定通過到來的聲音 localNote.soundName = UILocalNotificationDefaultSoundName; // 1.6.設定應用表徵圖左上方顯示的數字 localNote.applicationIconBadgeNumber = 999; // 1.7.設定一些額外的資訊 localNote.userInfo = @{@"qq" : @"704711253", @"msg" : @"success"}; // 2.執行通知 [[UIApplication sharedApplication] scheduleLocalNotification:localNote];}
效果如下:
3 取消通知
// 取消所有本地通知
[application cancelAllLocalNotifications];
三、遠程推送
與Android上我們自己實現的推送服務不一樣,Apple對裝置的控制非常嚴格,訊息推送的流程必須要經過APNs(Apple Push Notification service).
一般情況下如果一個程式退到後台就不能運行代碼(Audio、VoIP等等可以在後台運行),或者程式退出後,那麼它就和對應應用的後台伺服器斷開了連結,就收不到伺服器發送的資訊了,但是每台裝置只要連網就會和蘋果的APNs伺服器建立一個長串連(persistent IP connection),這樣只要通過蘋果的APNs伺服器,我們自己的伺服器就可以間接的和裝置保持串連了,示意圖如下:
使用步驟:
1 勾選Backgroud Modes -> Remote notifications,主要是iOS7之後,蘋果支援後台運行,如果這裡開啟後,當接收到遠程推送後,程式在後台也可以做一些處理,如下圖所示:
2 遠程推送的註冊與本地推送不同,iOS8.0前後也不同,代碼見下面。
另外,在第一次使用推送時,可能會有這樣的疑問:didFinishLaunchingWithOptions會在每次開啟程式時被調用,那是不是每次都會調用註冊函數,每次都會彈窗詢問使用者"是否允許推播通知"?其實這個視窗只會在第一次開啟程式時彈出一次,無論使用者允許或不允許蘋果會記住使用者的選擇,註冊函數調用多次對使用者也沒什麼影響
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // iOS8之後和之前應區別對待 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;}/// 這個函數存在的意義在於:當使用者在設定中關閉了通知時,程式啟動時會調用此函數,我們可以擷取使用者的設定- (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings { [application registerForRemoteNotifications];}
3 如果註冊失敗,比如沒有認證等等,會調用:
/// 註冊失敗調用- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error { NSLog(@"遠程通知註冊失敗:%@",error);}
4 擷取deviceToken
如果使用者同意,蘋果會根據應用的 bundleID 和 手機UDID 產生 deviceToken,然後調用 application 的 didregister 方法返回 devicetoken,程式應該把 devicetoken 發給應用的伺服器,伺服器有義務將其儲存(如果允許多重網路節點,可能存多個 devicetoken)。deviceToken也是會變的: ”If the user restores backup data to a new device or computer, or reinstalls the operating system, the device token changes“,因此應每次都發給伺服器(provider)
/// 使用者同意後,會調用此程式,擷取系統的deviceToken,應把deviceToken傳給伺服器儲存,此函數會在程式每次啟動時調用(前提是使用者允許通知)- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken { NSLog(@"deviceToken = %@",deviceToken);}
5 使用者點擊了通知
預設會開啟程式。處理代碼有三個函數,分iOS7之前之後和程式是否處於後台
5.1 iOS7及其之之後
此函數無論是程式被殺死還是處於後台,只要使用者點擊了通知,都會被調用,因此如果是iOS7,則不必在didFinishLaunchingWithOptions中做處理,只在下面函數做處理即可,此時應避免在didFinishLaunchingWithOptions函數中也做重複處理。
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler { // userInfo}
註:當在第一步開啟後台運行後,使用者不點擊通知,也可以執行:
- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo fetchCompletionHandler:(void(^)(UIBackgroundFetchResult))completionHandler
5.2 iOS7之前
當使用者點擊通知後,如果程式被殺死則會調用下面第一個函數,如果程式處於後台會調用下面第二個函數,因此下面兩個函數應搭配使用
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // 擷取遠程推送訊息 NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]; if (userInfo) { // 有推送的訊息,處理推送的訊息 } return YES;}/// iOS3之後才有,只有在程式處於後台時,使用者點擊了通知後才會被調用,應搭配didFinishLaunchingWithOptions使用- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo { // userInfo}
在實際編程時,如果想相容iOS7以前,三個函數可同時使用,都列出來,系統會自動選擇合適的調用。
6 總結下函數的調用:
首次安裝後啟動:
didRegisterForRemoteNotificationsWithDeviceToken 被調用
系統詢問使用者是否同意接收 Notifications
不管使用者選擇同意或拒絕,didRegisterUserNotificationSettings 被調用
應用非初次開機時:
如果 notifications 處於拒絕狀態:didRegisterUserNotificationSettings 被調用
如果 notifications 處於允許狀態
didRegisterForRemoteNotificationsWithDeviceToken 被調用
didRegisterUserNotificationSettings 被調用
應用運行過程中使用者修改 notifications 設定:
從拒絕變為允許:didRegisterForRemoteNotificationsWithDeviceToken 被調用
從允許變為拒絕:什麼也不發生
7 服務端推送的格式
{ "aps" : { // 必須有 "alert" : "string", "body" : "string", "badge" : number, "sound" : "string" }, "NotiId" : 20150821, // 自訂key值}
8 推送的大小限制
遠程通知負載的大小根據伺服器使用的API不同而不同。當使用HTTP/2 provider API時,負載最大為4kB;當使用legacy binary interface時,負載最大為2kB。當負載大小超過規定的負載大小時,APNs會拒絕發送此通知。
9 整體如下圖所示(以微信推送為例):
10 最後,還需要申請認證,這裡不再詳述-_-
本文已被整理到了《iOS推送教程》,歡迎大家學習閱讀。
以上就是本文的全部內容,希望對大家的學習有所協助,也希望大家多多支援雲棲社區。