iOS 推送通知流程

本文根据此文稍作精简: https://www.jianshu.com/p/d18c51b3321d
作者:Poker_Facer

一、本文总结了项目iOS8-10的通知适配工作

iOS推送通知几乎每年都在改,直到iOS10的出现,苹果对这块进行了重构。本文总结了一下项目中iOS8-10的通知模块适配工作。

二、实际开发中的适配工作

在实际开发中,iOS8-10的适配代码如下:

- (void)setupWithOptions:(NSDictionary *)launchOptions {
    // iOS8-10.0 注册通知
    if (!IS_IOS(10.0)) {
        UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil];
        [[UIApplication sharedApplication] registerUserNotificationSettings:settings];
    }
    // iOS10.0+ 开启通知权限
    else {
        UNUserNotificationCenter *center = [UNUserNotificationCenter currentNotificationCenter];
        center.delegate = self;
        UNAuthorizationOptions options = UNAuthorizationOptionBadge | UNAuthorizationOptionSound | UNAuthorizationOptionAlert;
        [center requestAuthorizationWithOptions:options completionHandler:^(BOOL granted, NSError * _Nullable error) {

        }];
    }
    
    // 启动极光推送SDK
    [JPUSHService setupWithOption:launchOptions
                           appKey:kJPUSHAppKey
                          channel:channel
                 apsForProduction:VERSION_RELEASE
            advertisingIdentifier:nil];
    
    // 极光推送日志打印开关
    if (VERSION_RELEASE) {
        [JPUSHService setLogOFF];
    } else {
        [JPUSHService setDebugMode];
    }
    
    // 注册极光推送(内部会调用registerForRemoteNotifications注册APNS)
    JPUSHRegisterEntity *entity = [[JPUSHRegisterEntity alloc] init];
    entity.types = JPAuthorizationOptionAlert | JPAuthorizationOptionBadge | JPAuthorizationOptionSound;
    [JPUSHService registerForRemoteNotificationConfig:entity delegate:self];
}

iOS10通过UNNotificationTrigger这个类的子类来区分远程通知和本地通知。

  • iOS8 通知请求权限修改。即UIRemoteNotificationType由UIUserNotificationType代替,并加入了UIUserNotificationSettings。

  • iOS9 通知加入可输入操作。

  • iOS10 通知重构,全新框架UNUserNotifications。

三、远程通知与本地通知的触发和调用

各版本通知中使用的类图表:


各版本通知类图.png

1、普通本地通知

  • 程序杀死状态。
    所有版本,本地通知在- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法中获取。[launchOptions objectForKey:UIApplicationLaunchOptionsLocalNotificationKey]即为我们需要的本地通知对象。没有通知时launchOptions为空。
  • 程序后台状态。
    iOS10以下,在点击通知会调用- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification方法。
    iOS10会调用- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler方法。
  • 程序前台状态。
    iOS10以下不会触发通知栏,并且还是调用- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification方法。
    iOS10会立马触发- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions))completionHandler方法,并且能根据completionHandler回调配置通知的显示样式。点击通知后同样会触发- (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(void (^)())completionHandler方法。

值得注意的是,程序后台状态下,代理方法都是在用户点击通知后才会执行,收到通知点击应用图标启动是不会走代理方法的。

程序杀死状态下启动应用只会触发- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions方法。

iOS10所有的通知,都只会触发下面两个方法。而区分远程还是本地通知,就是前面提到的UNNotificationTrigger的子类,当UNNotificationRequest的trigger为UNPushNotificationTrigger时,说明此通知为远程通知,否则为本地通知。下面为代理方法:

// iOS10通知代理方法,应用程序在前台的时候调用
- (void)userNotificationCenter:(UNUserNotificationCenter *)center willPresentNotification:(UNNotification *)notification withCompletionHandler:(void (^)(UNNotificationPresentationOptions options))completionHandler {
    UNNotificationRequest *request = notification.request;
    NSDictionary *userInfo = request.content.userInfo;
    
    if ([request.trigger isKindOfClass:[UNPushNotificationTrigger class]]) {
        NSLog(@"收到了一个远程推送:%@", userInfo);
    } else {
        NSLog(@"收到了一个本地推送:%@", userInfo);
    }
    
    //此方法回调,设置程序前台时banner提示框的显示选项
completionHandler(UNNotificationPresentationOptionAlert | UNNotificationPresentationOptionSound);
}

// iOS10通知代理方法,通知前后台点击时会触发。
-  (void)userNotificationCenter:(UNUserNotificationCenter *)center didReceiveNotificationResponse:(UNNotificationResponse *)response withCompletionHandler:(nonnull void (^)(void))completionHandler {
    NSDictionary *userInfo = response.notification.request.content.userInfo;
    // 只有本地通知才能获取到这个key对应的value
    NSData *modelData = userInfo[CYLocalNotificationModelKey];
    if (modelData) {
        [[CYLocalNotificationMgr sharedManager] handleLocalNotificationResponse:response];
    }

    completionHandler();
}

远程通知和本地通知之所以能统一规范,就是因为其原理类似,唯一的区别在于远程通知需要获取用户的token,通过苹果的APNS将通知发送给对应用户,而具体通知的内容则由后台配置的那个aps字典决定了。

2、普通远程通知

下面是iOS8-9远程通知相关的代理方法

/**
 已经收到远程推送消息
 @param userInfo 收到的userInfo信息
 */
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
    if (application.applicationState == UIApplicationStateActive) {
        NSLog(@"程序运行中收到通知");
    } else {
        NSLog(@"程序不活跃中收到通知");
    }
}

/**
 后台模式收到的远程推送信息,需要开启后台模式的远程推送,实现了此方法,上面的方法就失效。
 */
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
    if (application.applicationState == UIApplicationStateActive) {
        NSLog(@"程序运行中收到通知");
    } else {
        NSLog(@"程序不活跃中收到通知");
    }      
    completionHandler(UIBackgroundFetchResultNewData);
}

远程推送需要后台配置title等参数,也就是在本地通知中设置的那些什么alertTitle,alertBody啥子的。基本格式如下:

{
  "aps" : {
    "alert" : {
      "title" : "主标题",
      "subtitle" : "子标题",
      "body" : "通知内容"
    },
    "badge" : 2,
    "sound" : "alert.wav"
  },
  "custom" : "something"
}

其中aps为固定对象,后台需要传输自己的参数可以自行添加与aps并列的键值对。

  • subtitle是iOS10才有的,iOS10之前加不加无所谓。
  • sound为提示音,默认为default,自定义时别忘了后缀。
  • custom为自扩展字段,比如iOS10中图片和视频的连接地址就可以放这里。

3、高级远程通知

所谓高级,无非就是iOS10以下的可操作通知、iOS10中通知的增删改,以及UNNotificationServiceExtension和UNNotificationContentExtension这两个通知扩展了。由于是远程通知,这里仅附上测试aps格式,方便测试。

{
  "aps" : {
    "alert" : {
      "title" : "这是title",
      "subtitle" : "这是subtitle",
      "body" : "这是body"     
    },
    "mutable-content" : "1",            
    "category" : "myNotificationCategory"               
  }
}
  • 此aps可以用来测试上述所有高级远程通知。
  • category中的值必须与测试可操作通知时的category值一样。
  • 测试UNNotificationContentExtension时,必须和plist文件中的UNNotificationExtensionCategory所对应的值保持一致,编译对应的Scheme即可。
  • 凡是需要对通知内容做修改的都要加上mutable-content字段,UNNotificationServiceExtension中需要使用。例如下载通知中传来的图片或视频,先下载后再通知。
  • 通知的增删改都是根据request中的identifier进行查找和区分的。

通知其实也就这些,只是改的多,显得乱而已。理解了通知的流程与原理,处理这一模块的问题也就能得心应手。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容

  • 一、为什么会有这篇文章? 推送通知苹果几乎每年都改,但都是缝缝补补的功效,直到iOS10的出现,可能苹果觉得之前的...
    糖加一勺丶阅读 1,360评论 0 14
  • 极光推送: 1.JPush当前版本是1.8.2,其SDK的开发除了正常的功能完善和扩展外也紧随苹果官方的步伐,SD...
    Isspace阅读 6,715评论 10 16
  • 推送通知 注意:这里说的推送通知跟NSNotification有所区别 NSNotification是抽象的,不可...
    iOS开发攻城狮阅读 4,217评论 1 13
  • 推送通知注意:这里说的推送通知跟NSNotification有所区别NSNotification是抽象的,不可见的...
    醉叶惜秋阅读 1,511评论 0 3
  • 马云说:“西湖边上有个乞丐,面前放的不是钵不是碗,而是二维码。”
    一米阳光HUIWEI阅读 394评论 0 51