iOS推送通知原理

一、推送通知的基本原理:
苹果的推送服务通知是由自己专门的推送服务器APNs (Apple Push Notification service)来完成的,其过程是 APNs 接收到我们自己的应用服务器发出的被推送的消息,将这条消息推送到指定的 iOS 的设备上,然后再由 iOS设备通知到我们的应用程序,我们将会以通知或者声音的形式收到推送回来的消息。 iOS 远程推送的前提是,装有我们应用程序的 iOS 设备,需要向 APNs 服务器注册,注册成功后,APNs 服务器将会给我们返回一个 devicetoken,我们获取到这个 token 后会将这个 token 发送给我们自己的应用服务器。当我们需要推送消息时,我们的应用服务器将消息按照指定的格式进行打包,然后结合 iOS 设备的 devicetoken 一起发给 APNs 服务器。我们的应用会和 APNs 服务器维持一个基于 TCP 的长连接,APNs 服务器将新消息推送到iOS 设备上,然后在设备屏幕上显示出推送的消息。
设备注册APNs的流程图:


iOS推送通知原理
文章图片
推送流程图 上图完成了如下步骤:
1.Device(设备)连接APNs服务器并携带设备序列号(UUID)
2.连接成功,APNs经过打包和处理产生devicetoken并返回给注册的Device(设备)
3.Device(设备)携带获取的devicetoken发送到我们自己的应用服务器
4.完成需要被推送的Device(设备)在APNs服务器和我们自己的应用服务器的注册
推送过程图:


iOS推送通知原理
文章图片
Push 过程 1.首先,我们的设备安装了具有推送功能的应用(应用程序要用代码注册消息推动),我们的 iOS设备在有网络的情况下会连接APNs推送服务器,连接过程中,APNS 服务器会验证devicetoken,连接成功后维持一个基于TCP 的长连接;
2.Provider(我们自己的应用服务器)收到需要被推送的消息并结合被推送的 iOS设备的devicetoken一起打包发送给APNS服务器;
3.APNS服务器将推送信息推送给指定devicetoken的iOS设备;
4.iOS设备收到推送消息后通知我们的应用程序并显示和提示用户(声音、弹出框)
信息包结构图:


iOS推送通知原理
文章图片
信息结构图 上图显示的这个消息体就是我们的应用服务器(Provider)发送给APNs服务器的消息结构,APNs验证这个结构正确并提取其中的信息后,再将消息推送到指定的iOS设备。这个结构体包括五个部分,第一个部分是命令标示符,第二个部分是我们的devicetoken的长度,第三部分是我们的devicetoken字符串,第四部分是推送消 息体(Payload)的长度,最后一部分也就是真正的消息内容了,里面包含了推送消息的基本信息,比如消息内容,应用Icon右上角显示多少数字以及推送消息到达时所播放的声音等
Payload(消息体)的结构:
{
“aps”:{
“alert”:“send message”,
“badge”:1,
“sound”:“default” }
}
这其实就是个JSON结构体,alert标签的内容就是会显示在用户手机上的推送信息,badge显示的数量(注意是整型)是会在应用Icon右上角显示的数量,提示有多少条未读消息等,sound就是当推送信息送达是手机播放的声音,传defalut就标明使用系统默认声音。
二、配置证书 1.登录Apple的Developer Member Center、
打开Safari浏览器,地址栏输入并打开此网址:https://developer.apple.com/membercenter/index.action,然后用开发者账号登录.
选择点击Certificates, Identifiers & Profiles., 打开后在左侧列表中选择Certificates大类下面的All子类,出现如下界面:


iOS推送通知原理
文章图片
Apple dev 2.选择证书的类型,Sandbox或者Productuon
根据你的需求选择Apple Push Notification service SSL (Sandbox) 或者 Apple Push Notification service SSL (Sandbox & Production)类型的证书.
然后点击继续.
这里要注意一点,选择证书的正确类型非常重要:
一个用Development Provisioning Profile签名的App生成的 Device Token只能和Sandbox APNS Server结合使用(开发环境测试用).
一个用AdHoc Provisioning Profile或者AppStore Provisioning Profile签名的App生成的Device Token只能和Production APNS Server结合使用(生产环境发布使用).
提示:
有些用户已经报告过关于使用二合一的推送证书(Apple Push Notification service SSL (Sandbox & Production),新出的 )在调试测试环境进行推送时会出问题.


iOS推送通知原理
文章图片
生成推送证书 3.选择你对应的App ID
前提你要先已经创建了你App使用的App ID,并在App ID下拉选项中选择你对应的App ID.
点击继续按钮继续.


iOS推送通知原理
文章图片
选择对应AppleID 4.查看创建证书签名请求相关步骤信息
查看一下如何创建一个CSR文件的信息.
点击继续.


iOS推送通知原理
文章图片
apns-04 该页面有关于如何一步一步创建CSR文件的详细信息,下面我们就跟着一步一步来创建CSR文件.
5.打开钥匙串访问程序
打开Launchpad中的其他组的Keychain Access App,就是钥匙串访问程序.


iOS推送通知原理
文章图片
apns-05 6.创建证书签名请求文件
点击钥匙串访问程序的菜单栏
选择证书助理子菜单
选择从证书颁发机构请求证书...子子菜单


iOS推送通知原理
文章图片
apns-06 7.填写证书签名请求文件详细信息
输入email地址,为了安全和保险起见,最好填写和你Apple开发者账号对应的email地址.
接着输入常用名称.
最好选择存储到磁盘选项.
最后点击继续.
iOS推送通知原理
文章图片
apns-07 8.保存证书签名请求文件到磁盘
选择一个存储位置以及输入一个你喜欢的文件名称
点击保存


iOS推送通知原理
文章图片
apns-08 9.完成证书签名请求文件制作
点击完成


iOS推送通知原理
文章图片
apns-09 10.上传正确签名请求文件
继续回到第四步后的浏览器界面
选择 Choose File...来上传
在弹出的文件打开对话框中定位到刚刚存储的证书签名请求文件所在目录,并选择打开该CSR文件.
点击继续


iOS推送通知原理
文章图片
apns-10 11.下载证书文件
上一步继续之后,你的证书应该就已经生成好了.
点击下载按钮来下载证书.
打开下载的证书文件,一般会自动用钥匙串访问程序打开的.


iOS推送通知原理
文章图片
apns-11 三、代码配置 打开AppDelegate.m 文件,在- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 方法中添加下面代码,注册消息推送
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.
if (IOSVersion >= 10.0) {
UNUserNotificationCenter *center = [UNUserNotificationCentercurrentNotificationCenter];
// 必须写代理,不然无法监听通知的接收与点击
center.delegate=self;
[centerrequestAuthorizationWithOptions:(UNAuthorizationOptionAlert | UNAuthorizationOptionBadge | UNAuthorizationOptionSound) completionHandler:^(BOOL granted, NSError * _Nullable error) {
if(granted) {
// 点击允许
NSLog(@"注册成功");
[centergetNotificationSettingsWithCompletionHandler:^(UNNotificationSettings * _Nonnull settings) {
NSLog(@"%@", settings);
}];
}else{
// 点击不允许
NSLog(@"注册失败");
}
}];
}else{
[[UIApplication sharedApplication] registerUserNotificationSettings:[UIUserNotificationSettings settingsForTypes:(UIUserNotificationTypeSound | UIUserNotificationTypeAlert | UIUserNotificationTypeBadge) categories:nil]];
}
[[UIApplication sharedApplication] registerForRemoteNotifications];
}


#pragma mark Push methodsIniOS9
- (void)application:(UIApplication*)application didRegisterUserNotificationSettings:(UIUserNotificationSettings*)notificationSettings
{
[application registerForRemoteNotifications];
}
/** APP已经接收到“远程”通知(推送) - 透传推送消息*/
- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo fetchCompletionHandler:(void(^)(UIBackgroundFetchResultresult))completionHandler {
NSLog(@"\n>>>[Receive RemoteNotification - Background Fetch]:%@\n\n",userInfo);
completionHandler(UIBackgroundFetchResultNewData);
}
- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userinfo
{
NSLog(@"userInfo == %@",userInfo);
}
#pragma mark Push methodsIniOS10
- (void)userNotificationCenter:(UNUserNotificationCenter*)center willPresentNotification:(UNNotification*)notification withCompletionHandler:(void(^)(UNNotificationPresentationOptionsoptions))completionHandler{
completionHandler(UNNotificationPresentationOptionBadge|UNNotificationPresentationOptionSound|UNNotificationPresentationOptionAlert);
}
//App通知的点击事件
- (void)userNotificationCenter:(UNUserNotificationCenter*)center didReceiveNotificationResponse:(UNNotificationResponse*)response withCompletionHandler:(void(^)(void))completionHandler{
//收到推送的请求
UNNotificationRequest*request = response.notification.request;


//收到推送的内容
UNNotificationContent*content = request.content;


//收到用户的基本信息
NSDictionary*userInfo = content.userInfo;
completionHandler(); // 系统要求执行这个方法
}
下面方法是返回 ANPs 苹果推送服务器生成的唯一标识
/** 接收服务器传回的设备唯一标识 token */
- (void)application:(UIApplication*)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData*)deviceToken
{
NSString *token = [[deviceToken description] stringByTrimmingCharactersInSet:[NSCharacterSet characterSetWithCharactersInString:@"<>"]];
// 第一次运行获取到DeviceToken时间会比较长!
// 将deviceToken转换成字符串,以便后续使用
NSString *token = [deviceToken description];
NSLog(@"description %@", token);
}
/** 注册推送服务失败 */
- (void)application:(UIApplication*)application didFailToRegisterForRemoteNotificationsWithError:(NSError*)error
{
NSLog(@"注册失败 %@",error);
}
四、测试 测试推送链路是否通畅的工具有很多,大家自行选择,比如个推等等。。。需要注意的是推送是有丢失率的,不是100%每推必达!!!
【iOS推送通知原理】Demo地址

    推荐阅读