详解iOS本地推送与远程推送

一、简介

分为本地推送和远程推送2种。可以在应用没有打开甚至手机锁屏情况下给用户以提示。它们都需要注册,注册后系统会弹出提示框(如下图)提示用户是否同意,如果同意则正常使用;如果用户不同意则下次打开程序也不会弹出该提示框,需要用户到设置里面设置。一共有三种提示类型:

UIUserNotificationTypeBadge:应用图标右上角的信息提示

UIUserNotificationTypeSound:播放提示音

UIUserNotificationTypeAlert:提示框

二、本地推送

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 {
 // 这里添加处理代码
}

可以看到,处理代码有两个方法,一个是

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

效果如下:

3 取消通知

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

三、远程推送

与Android上我们自己实现的推送服务不一样,Apple对设备的控制非常严格,消息推送的流程必须要经过APNs(Apple Push Notification service).

一般情况下如果一个程序退到后台就不能运行代码(Audio、VoIP等等可以在后台运行),或者程序退出后,那么它就和对应应用的后台服务器断开了链接,就收不到服务器发送的信息了,但是每台设备只要联网就会和苹果的APNs服务器建立一个长连接(persistent IP connection),这样只要通过苹果的APNs服务器,我们自己的服务器就可以间接的和设备保持连接了,示意图如下:

使用步骤:

1 勾选Backgroud Modes -> Remote notifications,主要是iOS7之后,苹果支持后台运行,如果这里打开后,当接收到远程推送后,程序在后台也可以做一些处理,如下图所示:

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 整体如下图所示(以微信推送为例):

10 最后,还需要申请证书,这里不再详述-_-

本文已被整理到了《iOS推送教程》,欢迎大家学习阅读。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • iOS10 推送最新特性研究

    最近在研究iOS10关于推送的新特性, 相比之前确实做了很大的改变,总结起来主要是以下几点: 1.推送内容更加丰富,由之前的alert 到现在的title, subtitle, body  2.推送统一由trigger触发  3.可以为推送增加附件,如图片.音频.视频,这就使推送内容更加丰富多彩  4.可以方便的更新推送内容 import 新框架 添加新的框架 UserNotifications.framework #import <UserNotifications/UserNotificat

  • iOS Remote Notification远程消息推送处理

    远程消息推送处理场景有三种:分别是app还没有运行.app在前台运行以及app在后台运行,下面介绍相关流程及三种场景下处理步骤 1.流程 (1)注册通知 首先是在注册远程消息推送,需要注意的是iOS8及以后的系统中注册方法有所改变(同时证书设置以及push权限也需要开启).这一步的目的是,允许app接收远程消息推送. (2)绑定deviceToken deviceToken相当于设备的一个标识,服务器根据这个标识来进行消息推送. ①当用户同意app接收远程消息推送后,手机会向APNs发起一个请求

  • iOS开发之(APNS)远程推送实现代码 附证书与真机调试

    远程推送通知 什么是远程推送通知 顾名思义,就是从远程服务器推送给客户端的通知(需要联网)远程推送服务,又称为APNs(ApplePush Notification Services) 为什么需要远程推送通知传统获取数据的局限性只要用户关闭了app,就无法跟app的服务器沟通,无法从服务器上获得最新的数据内容远程推送通知可以解决以上问题不管用户打开还是关闭app,只要联网了,都能接收到服务器推送的远程通知远程推送通知使用须知所有的苹果设备,在联网状态下,都会与苹果的服务器建立长连接什么是长连接只

  • iOS远程推送Push开发教程

    远程推送通知 什么是远程推送通知 顾名思义,就是从远程服务器推送给客户端的通知(需要联网)远程推送服务,又称为APNs(Apple Push Notification Services) 为什么需要远程推送通知 传统获取数据的局限性 只要用户关闭了app,就无法跟app的服务器沟通,无法从服务器上获得最新的数据内容 远程推送通知可以解决以上问题 不管用户打开还是关闭app,只要联网了,都能接收到服务器推送的远程通知 远程推送通知使用须知 所有的苹果设备,在联网状态下,都会与苹果的服务器建立长连接

  • 轻松搞定iOS远程消息推送

    一.引言 IOS中消息的推送有两种方式,分别是本地推送和远程推送,本地推送在http://www.jb51.net/article/93602.htm这篇博客中有详细的介绍,这里主要讨论远程推送的流程与配置过程. 二.远程推送机制的原理 1.从一张很火的图说起 搜索IOS远程推送,你总能看到一张如下的流程示意图,因为这张图确实很火,所以我也将它引用在此: 这张图示意的很清晰,大致意思是这样:你的应用服务端将消息发送到apple的APNS服务器,APNS服务器将消息推送到指定的Iphone,最后由

  • iOS10推送通知开发教程

    虽然通知经常被过度使用,但是通知确实是一种获得用户关注和通知他们需要更新或行动的有效方式.iOS 10有了新的通知,如新消息.商业信息和时间表的变化.在本教程中,我将向你展示如何使用通知在你的iOS应用程序,并且显示iOS 10引入了新特性.开发iOS 10推送通知你需要最新版本的Xcode,Xcode 8测试版,这些目前都是可下载的,在下载页面. 你可以去Github下载本教程的整个工程. 开始 在Xcode中启用推送通知是很容易的,但你需要几个步骤. 创建一个新的工程,给它起一个唯一的Bun

  • iOS消息远程推送通知

    本文实例为大家分享了iOS消息推送.iOS远程通知代码,供大家参考,具体内容如下 消息推送 /* 要开发测试消息机制的程序,必须用真机测试 推送消息的类型 UIRemoteNotificationTypeNone 不接收推送消息 UIRemoteNotificationTypeBadge 接收图标数字 UIRemoteNotificationTypeSound 接收音频 UIRemoteNotificationTypeAlert 接收消息文字 UIRemoteNotificationTypeNe

  • iOS10全新推送功能实现代码

    从iOS8.0开始推送功能的实现在不断改变,功能也在不断增加,iOS10又出来了一个推送插件的开发(见最后图),废话不多说直接上代码: #import <UserNotifications/UserNotifications.h> - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // Override point for

  • iOS实现远程推送原理及过程

    推送通知,是现在的应用必不可少的功能.那么在 iOS 中,我们是如何实现远程推送的呢?iOS 的远程推送原理又是什么呢?在做 iOS 远程推送时,我们会遇到各种各样的问题.那么首先让我们准备一些做推送需要的东西.我们需要一个付费的苹果开发者账号(免费的不可以做远程推送),有了开发者账号,我们可以去苹果开发者网站,配置自己所需要的推送的相关证书.然后下载证书,供我们后面使用,详细的证书配置过程,我们下面再说. 首先我们要说说iOS推送通知的基本原理: 苹果的推送服务通知是由自己专门的推送服务器AP

  • iOS10 适配远程推送功能实现代码

    iOS10正式版发布之后,网上各种适配XCode8以及iOS10的文章满天飞.但对于iOS10适配远程推送的文章却不多.iOS10对于推送的修改还是非常大的,新增了UserNotifications Framework,今天就结合自己的项目,说一说实际适配的情况. 一.Capabilities中打开Push Notifications 开关 在XCode7中这里的开关不打卡,推送也是可以正常使用的,但是在XCode8中,这里的开关必须要打开,不然会报错: Error Domain=NSCocoa

随机推荐