@(〓〓 iOS-实用技术)[远程/本地推送]
- 作者: Liwx
- 邮箱: 1032282633@qq.com
目录
- 04.iOS远程推送通知 APNs远程推送,极光推送
- 1.远程推送通知
- 远程推送的原理
- 2.远程推送通知实现, 客户端需要做的事
- 1.需要配置真机调试证书, 推送测试证书
- 2.请求苹果获得deviceToken
- 3.远程推送服务端配置
- PushMeBaby的配置
- 3.iOS客户端监听用户对通知的点击
- 监听远程推送通知点击的两种方法
- 4.APNs远程推送实现的具体步骤
- 配置iOS应用的环境配置
- APNs远程推送通知实现
- PushMeBaby的配置
- 运行PushMeBaby和iPhone客户端程序
- 5.极光推送
- 使用第三方推送的好处
- 极光推送介绍
- 极光推送的集成
1.远程推送通知
远程推送通知
必须要真机调试
,因为真机才有UDID
,才能够生成deviceToken
.应用的Bundle ID必须在苹果开发者中心创建,并且生成真机调试证书
(.cer文件或.p12文件),推送通知调试证书
(.cer文件或.p12文件)和profile描述文件(.mobileprovision文件).如果是要打包测试和发布
,必须生成打包测试推送通知证书和发布推送通知证书.
- 运行真机调试证书和推送通知调试证书会在keychain生成证书
远程推送的原理
-
所有苹果设备, 在联网状态下,都会与苹果服务器建立一个长连接
“长连接”: 相对的一个概念是”短连接”
- “长连接”
优势: 服务器可以向客户端发送信息,保证数据即时性
劣势: 占用客户端和服务器资源
- “短连接”
优势: 节省资源,一个会话结束后,立即释放资源
劣势: 服务器无法主动向客户端发信息
-
苹果设备“长连接”作用:
- 时间校准
- 系统升级
- 查找我的iPhone
- 推送通知..
原理就是借助苹果设备与APNs服务器之间的长连接, 借助APNs服务器将消息发送给客户端
案例解释
微信/QQ的消息发送机制
2.远程推送通知实现, 客户端需要做的事
1.需要配置真机调试证书, 推送测试证书
注意
: 必须要有付费版的开发者帐号才能配置生成证书.
配置证书参考教程: http://docs.jpush.io/client/ios_tutorials/#ios-8-uiusernotificationsettings
2.请求苹果获得deviceToken
注意: 只有真机可以调试推送通知
因为只有真机具备UDID, 才能够生成deviceToken
- 1.想苹果请求deviceToken
- iOS 8.0之后必须请求授权
- 获取当前设备的UDID, 以及, app bundle id , 向苹果服务器发送请求, 获取deviceToken
// ----------------------------------------------------------------------------
// 远程推送APNs必须使用真机调试,并且必须使用付费的开发者账号配置好对应的bundle ID和真机推送证书
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 向苹果请求DeviceToken
[self requestDeviceToken];
return YES;
}
// ------------------------------------------------------------------------
// 向苹果请求DeviceToken
- (void)requestDeviceToken
{
// iOS 8.0之后必须请求授权
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) {
// 1.请求授权
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:setting];
// 2.自动获取当前设备的UDID, 以及, app bundle id , 向苹果服务器发送请求, 获取deviceToken
[[UIApplication sharedApplication] registerForRemoteNotifications];
} else {
// iOS 8.0之前不用请求授权
// 2.自动获取当前设备的UDID, 以及, app bundle id , 向苹果服务器发送请求, 获取deviceToken
[[UIApplication sharedApplication] registerForRemoteNotificationTypes:UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert];
}
}
- 2.获取APNs返回的deviceToken
- 当远程通知注册后,APNs会通过调用这个方法,返回对应的deviceToken
// ----------------------------------------------------------------------------
// 当我们调用对应的api, 发送请求, 获取deviceToken , 苹果服务器响应之后, 就会调用这个方法, 把deviceToken 传给我们
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
// 获取到DeviceToken时,将deviceToken值写入到第三方APNs推送服务器PushMeBaby(使用Socket编写的调试服务器),可以在GitHub上下载
NSLog(@"%@", deviceToken);
}
3.远程推送服务端配置
在获得苹果返回的deviceToken后,发送deviceToken给公司的服务器.此处使
用PushMeBaby
模拟公司的服务器.PushMeBaby可以在GitHub上下载.
下载链接: https://github.com/stefanhafeneger/PushMeBaby
PushMeBaby的配置
- PushMeBaby设置
- 编译会出现错误, 直接把错误行注释即可;
- 需要填写客户端想苹果请求对应的deviceToken;
- 需要将开发者中心生成的推送真机测试证书改名为aps.cer 拖入PushMeBaby项目中;
- 编译成功后,运行效果图
- PushMeBaby设置deviceToken的两种方式
3.iOS客户端监听用户对通知的点击
监听远程推送通知点击的两种方法
- 1.用户点击通知才能调用的方法
// ----------------------------------------------------------------------------
// 监听远程推送通知点击(优先级较低),接收到通知,用户点击进入才会调用该方法
/**
当接收到推送通知之后, 并且满足一下条件
app 在前台时 ,会调用该方法
app 从后台进入到前台 (App一开始在后台, app 锁屏),点击通知会调用该方法
app 完全退出
如果app 完全退出, 这时候,如果用户点击通知, 打开APP , 不会调用这个方法
*/
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo;
- 2.一接收到通知就会立即调用该方法
- 效果: 当用户收到通知之后, 即使没有点击也会调用这个方法
- 条件:
- 1.需要
勾选后台模式 remote notification
,在后台 - 2.必须保证发送的推送通知格式, 包括
"content-available":"随便传"
- 3.
执行completionHandler
回调代码块
- 1.需要
// ----------------------------------------------------------------------------
// 监听远程推送通知点击(优先级较高),一接收到通知就会立即调用该方法
/**
当接收到推送通知之后, 并且满足一下条件
当我们实现, 这个方法时, 上面一个方法不再执行
计时APP 完全退出, 也会调用这个方法
completionHandler : 统计我们处理的时间, 耗电量, 刷新预览图片
{"aps" :
{
"alert" : "This is some fancy message.",
"badge":1,
"content-available":"随便传"
}
}
*/
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler;
-
执行completionHandler 作用
1> 系统会估量App消耗的电量,并根据传递的UIBackgroundFetchResult 参数记录新数据是否可用
2> 调用完成的处理代码时,应用的界面缩略图会自动更新- 如果想要接收到通知后,不要用户点击通知, 就执行以下代码, 那么必须有三个要求:
1> 必须勾选后台模式Remote Notification
;
2> 告诉系统是否有新的内容更新(执行完成代码块)
3> 服务器发送通知的格式必须要有content-available字段("content-available":"随便传"
)
- 如果想要接收到通知后,不要用户点击通知, 就执行以下代码, 那么必须有三个要求:
4.APNs远程推送实现的具体步骤
配置iOS应用的环境配置
- 1.安装从开发者中心生成的真机调试证书,推送调试证书和描述文件profile
安装证书和描述文件后的效果图
- 2.配置Bundle ID必须要和开发者中心配置的Bundle ID一致
- 3.设置开启应用的后台模式(要实现一接收到通知就会立即调用的功能必须配置后台模式)
- 4.配置开发者真机调试
- 选择Build Settings,搜索code signing
APNs远程推送通知实现
- 1.iPhone手机响APNs远程推送服务器请求deviceToken
// ----------------------------------------------------------------------------
// 远程推送APNs必须使用真机调试,并且必须使用付费的开发者账号配置好对应的bundle ID和真机推送证书
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 向苹果请求DeviceToken
[self requestDeviceToken];
return YES;
}
// ------------------------------------------------------------------------
// 向苹果请求DeviceToken
- (void)requestDeviceToken
{
// iOS 8.0之后必须请求授权
if ([UIDevice currentDevice].systemVersion.floatValue >= 8.0) {
// 1.请求授权
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:setting];
// 2.自动获取当前设备的UDID, 以及, app bundle id , 向苹果服务器发送请求, 获取deviceToken
[[UIApplication sharedApplication] registerForRemoteNotifications];
} else {
// iOS 8.0之前不用请求授权
// 2.自动获取当前设备的UDID, 以及, app bundle id , 向苹果服务器发送请求, 获取deviceToken
[[UIApplication sharedApplication] registerForRemoteNotificationTypes:UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert];
}
}
- 2.监听APNs苹果服务器响应的deviceToken(目的是为了配置PushMeBaby中的DeviceToken)
// ----------------------------------------------------------------------------
// 当我们调用对应的api, 发送请求, 获取deviceToken , 苹果服务器响应之后, 就会调用这个方法, 把deviceToken 传给我们
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
// 获取到DeviceToken时,将deviceToken值写入到第三方APNs推送服务器PushMeBaby(使用Socket编写的调试服务器),可以在GitHub上下载
NSLog(@"%@", deviceToken);
}
// ----------------------------------------------------------------------------
// 当获取DeviceToken失败是会调用该方法
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error
{
NSLog(@"注册失败: %@", error.localizedDescription);
}
- 3.客户端监听远程推送通知的点击
// ----------------------------------------------------------------------------
// 监听远程推送通知点击(优先级较低),接收到通知,用户点击进入才会调用该方法
/**
当接收到推送通知之后, 并且满足一下条件
app 在前台时 ,会调用该方法
app 从后台进入到前台 (App一开始在后台, app 锁屏),点击通知会调用该方法
app 完全退出
如果app 完全退出, 这时候,如果用户点击通知, 打开APP , 不会调用这个方法
*/
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
NSLog(@"接收到远程推送通知");
// UISwitch *sw = [[UISwitch alloc] init];
// [self.window.rootViewController.view addSubview:sw];
}
// ----------------------------------------------------------------------------
// 监听远程推送通知点击(优先级较高),一接收到通知就会立即调用该方法
/**
当接收到推送通知之后, 并且满足一下条件
当我们实现, 这个方法时, 上面一个方法不再执行
计时APP 完全退出, 也会调用这个方法
completionHandler : 统计我们处理的时间, 耗电量, 刷新预览图片
效果: 当用户收到通知之后, 即使没有点击也会调用这个方法
条件: 1 .需要勾选后台模式 remote notification,在后台
2. 必须保证发送的推送通知格式, 包括 "content-available":"随便传"
3. 执行completionHandler 回调代码块
{"aps" :
{
"alert" : "This is some fancy message.",
"badge":1,
"content-available":"随便传"
}
}
*/
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
{
NSLog(@"fetchCompletionHandler: 接收到远程推送通知");
UISwitch *sw = [[UISwitch alloc] init];
[self.window.rootViewController.view addSubview:sw];
completionHandler(UIBackgroundFetchResultNewData);
}
@end
PushMeBaby的配置
1.如果PushMeBaby编译出现错误, 直接把错误行注释即可;
2.需要填写iPhone客户端向苹果请求对应的deviceToken(两种方式任选一种)
-
3.如果要让iOS应用接收到通知立即执行响应操作
- 必须在PushMeBaby中将payload数据内容添加
"content-available":"随便传"
- 必须在PushMeBaby中将payload数据内容添加
需要将开发者中心生成的推送真机测试证书改名为aps.cer 拖入PushMeBaby项目中参与编译;
运行PushMeBaby和iPhone客户端程序
1.运行iPhone客户端程序,并退到后台
-
2.运行PushMeBaby,确保和iPhone客户端的deviceToken一致
- 点击Push发送通知到客户端
3.此时客户端会在顶部弹出通知.APNs远程推送测试成功.
5.极光推送
第三方推送: 极光推送,个推,信鸽...
第三方服务合集: http://mdsa.51cto.com/services/
使用第三方推送的好处
- 使用第三方推送的好处
优势
减少开发及维护成本:
应用开发者不需要去开发维护自己的推送服务器与 APNs 对接。
集成了 JPush iOS SDK 后不必自己维护更新 device token。
通过 JPush 的 Web Portal 直接推送,也可以调用JPush的 HTTP 协议 API 来完成,开发工作量大大减少。
减少运营成本:
极光推送支持一次推送,同时向 Android, iOS, WinPhone 三个平台。支持统一的 API 与推送界面。
极光推送提供标签、别名绑定机制,以及提供了非常细分的用户分群方式,运营起来非常简单、直观。
提供应用内推送:
除了使得 APNs 推送更简单,也另外提供应用内消息推送。这在类似于聊天的场景里很有必要。
极光推送介绍
- 极光推送发送通知流程图
极光推送的集成
- JPush iOS SDK 集成指南
- JPush iOS SDK 教程
- JPush SDK下载