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

MadSankt 8年前發布 | 40K 次閱讀 IOS iOS開發 移動開發

一、簡介

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

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

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

二、本地推送

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

    可以看到,處理代碼有兩個方法,一個是

    - (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]; }</code></pre>

    效果如下:

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

    3 取消通知

    // 取消所有本地通知
    [application cancelAllLocalNotifications];

    三、遠程推送    

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

    一般情況下如果一個程序退到后臺就不能運行代碼(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]; }</code></pre>

    3 如果注冊失敗,比如沒有證書等等,會調用:

    /// 注冊失敗調用
  • (void)application:(UIApplication )application didFailToRegisterForRemoteNotificationsWithError:(NSError )error { NSLog(@"遠程通知注冊失敗:%@",error); }</code></pre>

    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); }</code></pre>

    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 }</code></pre>

    在實際編程時,如果想兼容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 最后,還需要申請證書,這里不再詳述-_-


     

    文/時間已靜止(簡書)
     

 本文由用戶 MadSankt 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!