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

發表於2016-05-23

一、簡介

分為本地推送和遠端推送2種。可以在應用沒有開啟甚至手機鎖屏情況下給使用者以提示。它們都需要註冊,註冊後系統會彈出提示框(如下圖)提示使用者是否同意,如果同意則正常使用;如果使用者不同意則下次開啟程式也不會彈出該提示框,需要使用者到設定裡面設定。一共有三種提示型別:

  • UIUserNotificationTypeBadge:應用圖示右上角的資訊提示
  • UIUserNotificationTypeSound:播放提示音
  • UIUserNotificationTypeAlert:提示框
iOS推送——本地推送與遠端推送詳解(一圖看懂)

二、本地推送

1 註冊與處理

程式碼如下:

可以看到,處理程式碼有兩個方法,一個是
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification;
另一個是
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
如果程式沒有被殺死,即處於前臺或者後臺,那麼呼叫前者;如果程式被殺死,則呼叫後者。

2 傳送通知

程式碼如下

效果如下:

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

3 取消通知

三、遠端推送

與Android上我們自己實現的推送服務不一樣,Apple對裝置的控制非常嚴格,訊息推送的流程必須要經過APNs(Apple Push Notification service).

一般情況下如果一個程式退到後臺就不能執行程式碼(Audio、VoIP等等可以在後臺執行),或者程式退出後,那麼它就和對應應用的後臺伺服器斷開了連結,就收不到伺服器傳送的資訊了,但是每臺裝置只要聯網就會和蘋果的APNs伺服器建立一個長連線(persistent IP connection),這樣只要通過蘋果的APNs伺服器,我們自己的伺服器就可以間接的和裝置保持連線了,示意圖如下:

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

使用步驟:

1 Xcode設定

勾選Backgroud Modes -> Remote notifications,主要是iOS7之後,蘋果支援後臺執行,如果這裡開啟後,當接收到遠端推送後,程式在後臺也可以做一些處理,如下圖所示:

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

2 遠端推送的註冊與本地推送不同,iOS8.0前後也不同,程式碼見下面。

另外,在第一次使用推送時,可能會有這樣的疑問:didFinishLaunchingWithOptions會在每次開啟程式時被呼叫,那是不是每次都會呼叫註冊函式,每次都會彈窗詢問使用者”是否允許推送通知”?其實這個視窗只會在第一次開啟程式時彈出一次,無論使用者允許或不允許蘋果會記住使用者的選擇,註冊函式呼叫多次對使用者也沒什麼影響

3 如果註冊失敗,比如沒有證書等等,會呼叫:

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)

5 使用者點選了通知

預設會開啟程式。處理程式碼有三個函式,分iOS7之前之後和程式是否處於後臺

  • 5.1 iOS7及其之之後

此函式無論是程式被殺死還是處於後臺,只要使用者點選了通知,都會被呼叫,因此如果是iOS7,則不必在didFinishLaunchingWithOptions中做處理,只在下面函式做處理即可,此時應避免在didFinishLaunchingWithOptions函式中也做重複處理。

注:當在第一步開啟後臺執行後,使用者不點選通知,也可以執行: - (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo fetchCompletionHandler:(void(^)(UIBackgroundFetchResult))completionHandler

  • 5.2 iOS7之前

當使用者點選通知後,如果程式被殺死則會呼叫下面第一個函式,如果程式處於後臺會呼叫下面第二個函式,因此下面兩個函式應搭配使用

在實際程式設計時,如果想相容iOS7以前,三個函式可同時使用,都列出來,系統會自動選擇合適的呼叫。

6 總結下函式的呼叫:

首次安裝後啟動:

  • didRegisterForRemoteNotificationsWithDeviceToken 被呼叫
  • 系統詢問使用者是否同意接收 Notifications
  • 不管使用者選擇同意或拒絕,didRegisterUserNotificationSettings 被呼叫

應用非首次啟動時:

  • 如果 notifications 處於拒絕狀態:didRegisterUserNotificationSettings 被呼叫
  • 如果 notifications 處於允許狀態
    • didRegisterForRemoteNotificationsWithDeviceToken 被呼叫
    • didRegisterUserNotificationSettings 被呼叫
  • 應用執行過程中使用者修改 notifications 設定:
    • 從拒絕變為允許:didRegisterForRemoteNotificationsWithDeviceToken 被呼叫
    • 從允許變為拒絕:什麼也不發生

7 服務端推送的格式

8 推送的大小限制

遠端通知負載的大小根據伺服器使用的API不同而不同。當使用HTTP/2 provider API時,負載最大為4kB;當使用legacy binary interface時,負載最大為2kB。當負載大小超過規定的負載大小時,APNs會拒絕傳送此通知。

9 整體如下圖所示(以微信推送為例):

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

10 最後,還需要申請證書,這裡不再詳述-_-

相關文章