iOS本地推送與遠程推送詳解

1、簡介

分爲本地推送和遠程推送2種。能夠在應用沒有打開甚至手機鎖屏狀況下給用戶以提示。它們都須要註冊,註冊後系統會彈出提示框(以下圖)提示用戶是否贊成,若是贊成則正常使用;若是用戶不一樣意則下次打開程序也不會彈出該提示框,須要用戶到設置裏面設置。一共有三種提示類型:編程

  • UIUserNotificationTypeBadge:應用圖標右上角的信息提示    
  • UIUserNotificationTypeSound:播放提示音
  • UIUserNotificationTypeAlert:提示框

乾貨——iOS本地推送與遠程推送詳解(一圖看懂)

2、本地推送

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 { // 這裏添加處理代碼 }

能夠看到,處理代碼有兩個方法,一個是app

- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification;
另外一個是
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
若是程序沒有被殺死,即處於前臺或者後臺,那麼調用前者;若是程序被殺死,則調用後者。ide

 

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]; }

效果以下:fetch

乾貨——iOS本地推送與遠程推送詳解(一圖看懂)

3 取消通知ui

// 取消全部本地通知 [application cancelAllLocalNotifications];

3、遠程推送    

與Android上咱們本身實現的推送服務不同,Apple對設備的控制很是嚴格,消息推送的流程必需要通過APNs(Apple Push Notification service).spa

通常狀況下若是一個程序退到後臺就不能運行代碼(Audio、VoIP等等能夠在後臺運行),或者程序退出後,那麼它就和對應應用的後臺服務器斷開了連接,就收不到服務器發送的信息了,可是每臺設備只要聯網就會和蘋果的APNs服務器創建一個長鏈接(persistent IP connection),這樣只要經過蘋果的APNs服務器,咱們本身的服務器就能夠間接的和設備保持鏈接了,示意圖以下:

乾貨——iOS本地推送與遠程推送詳解(一圖看懂)

使用步驟:

1 勾選Backgroud Modes -> Remote notifications,主要是iOS7以後,蘋果支持後臺運行,若是這裏打開後,當接收到遠程推送後,程序在後臺也能夠作一些處理,以下圖所示:

乾貨——iOS本地推送與遠程推送詳解(一圖看懂)

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 總體以下圖所示(以微信推送爲例):

乾貨——iOS本地推送與遠程推送詳解(一圖看懂)

10 最後,還須要申請證書,這裏再也不詳述-_-

相關文章
相關標籤/搜索