关于iOS的Push Notification的响应

在说Push Notification的响应之前,先来讨论下iOS应用程序的状态,回调方法以及状态切换

  1. 应用程序的状态
状态 说明 描述
Not Running 未运行 程序没启动
Inactive 未激活 程序在前台运行,不过没有接收到事件。在没有事件处理情况下程序通常停留在这个状态
Active 激活 程序在前台运行而且接收到了事件。这也是前台的一个正常的模式
Backgroud 后台 程序在后台而且能执行代码,大多数程序进入这个状态后会在在这个状态上停留一会。时间到之后会进入挂起状态(Suspended)。有的程序经过特殊的请求后可以长期处于Backgroud状态
Suspended 挂起 程序在后台不能执行代码。系统会自动把程序变成这个状态而且不会发出通知。当挂起时,程序还是停留在内存中的,当系统内存低时,系统就把挂起的程序清除掉,为前台程序提供更多的内存。
  1. 回调方法

    • application:didFinishLaunchingWithOptions:
      - 本地通知:UIApplicationDidFinishLaunchingNotification
      - 触发时机:程序启动并进行初始化的时候后。
      - 适宜操作:这个阶段应该进行根视图的创建。
    • applicationDidBecomeActive:
      - 本地通知:UIApplicationDidBecomeActiveNotification
      - 触发时机:程序进入前台并处于活动状态时调用。
      - 适宜操作:这个阶段应该恢复UI状态(例如游戏状态)。
    • applicationWillResignActive:
      - 本地通知:UIApplicationWillResignActiveNotification
      - 触发时机:从活动状态进入非活动状态。
      - 适宜操作:这个阶段应该保存UI状态(例如游戏状态)。
    • applicationDidEnterBackground:
      - 本地通知:UIApplicationDidEnterBackgroundNotification
      - 触发时机:程序进入后台时调用。
      - 适宜操作:这个阶段应该保存用户数据,释放一些资源(例如释放数据库资源)。
    • applicationWillEnterForeground:
      - 本地通知:UIApplicationWillEnterForegroundNotification
      - 触发时机:程序进入前台,但是还没有处于活动状态时调用。
      - 适宜操作:这个阶段应该恢复用户数据。
    • applicationWillTerminate:
      - 本地通知:UIApplicationWillTerminateNotification
      - 触发时机:程序被杀死时调用。
      - 适宜操作:这个阶段应该进行释放一些资源和保存用户数据。
  2. 状态切换

    • 启动程序
      Not running =》Inactive =》Active
      • willFinishLaunchingWithOptions
      • didFinishLaunchingWithOptions
      • applicationDidBecomeActive
    • 按下home键
      Active =》Inactive =》Background =》(Suspended =》Not Running)
      • applicationWillResignActive
      • applicationDidEnterBackground
      • applicationWillTerminate:
    • 双击home键,再打开程序
      • applicationWillEnterForeground
      • applicationDidBecomeActive

言归正传,来说说Push Notification的响应。首先说iOS 10 之前。回调函数有两个,分别是

  • application:didReceiveRemoteNotification:
  • application:didReceiveRemoteNotification:fetchCompletionHandler:

这两个函数有什么区别呢?
简单来说,iOS7以上,你只需要实现第二个函数即可。因为如果你两个函数都实现的化,程序只会调用第二个。(相信现在也没有什么iOS不支持iOS7了吧。)
具体的请参考苹果文档
https://developer.apple.com/documentation/uikit/uiapplicationdelegate/1623117-application
https://developer.apple.com/documentation/uikit/uiapplicationdelegate/1623013-application

APP状态 消息推送 用户操作 是否触发
Not Running 收到推送消息提示 点击消息 YES
Not Running 收到推送消息提示 点击APP NO
Background 收到推送消息提示 点击消息 YES
Background 收到推送消息提示 点击APP NO
Active 无推送消息提示 YES

那么什么情况下application: didReceiveRemoteNotification: fetchCompletionHandler: 会被触发?

APP状态 消息推送 用户操作 是否触发
Not Running 收到推送消息提示 点击消息 YES
Not Running 收到推送消息提示 点击APP NO
Background 收到推送消息提示 点击消息 YES
Background 收到推送消息提示 点击APP NO
Active 无推送消息提示 YES

结论:如果应用不在激活状态,点击APP激活应用是不会触发application:didReceiveRemoteNotification:fetchCompletionHandler函数的,只有点击推送过来的消息,才会触发函数调用。

此外,在iOS7之后,不需要再考虑在应用未启动状态下,接到推送消息,在application: didFinishLaunchingWithOptions:中获取推送信息了。因为application:didReceiveRemoteNotification:fetchCompletionHandler在这种情况下会被调用。如果还继续在application: didFinishLaunchingWithOptions:中做推送处理,那么就会出现重复。

再看iOS10的情况,iOS10提供了两个delegate函数,分别对应应用处于激活状态和非激活状态

  • userNotificationCenter:willPresentNotification:withCompletionHandler:
  • userNotificationCenter:didReceiveNotificationResponse:withCompletionHandler:
APP状态 消息推送 用户操作 是否触发
Not Running 收到推送消息提示 点击消息 触发 didReceiveNotificationResponse
Not Running 收到推送消息提示 点击APP NO
Background 收到推送消息提示 点击消息 触发 didReceiveNotificationResponse
Background 收到推送消息提示 点击APP NO
Active 可自定义 触发willPresentNotification

触发willPresentNotification可自定义是否弹出消息提示,此时,若点击消息提示,则会调用didReceiveNotificationResponse。

综上,正确处理badge更新的好时机

  • 程序不在前台

    • 我们可以在applicationDidBecomeActive函数中从服务器获取各个badge的信息,并进行UI更新。
    • application:didReceiveRemoteNotification:fetchCompletionHandler 和application:didReceiveRemoteNotification:fetchCompletionHandler函数内仅处理对应页面的跳转。
  • 程序在前台
    application:didReceiveRemoteNotification:fetchCompletionHandler 和userNotificationCenter:willPresentNotification:withCompletionHandler函数内从服务器获取各个badge的信息,并进行UI更新。但不需要做对应页面的跳转。不希望打断用户当前的行为。

参考: iOS应用程序生命周期(前后台切换,应用的各种状态)详解

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

推荐阅读更多精彩内容