iOS 远程推送
远程推送代码实现
在iOS7下
- 注册远程通知
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// UIRemoteNotificationTypeBadge = 1 << 0, 应用图标右上角数字
// UIRemoteNotificationTypeSound = 1 << 1, 声音
// UIRemoteNotificationTypeAlert = 1 << 2, 提示
// UIRemoteNotificationTypeNewsstandContentAvailability = 1 << 3, 新闻,国内几乎没人使用
// 远程通知类型
UIRemoteNotificationType remoteTypes = UIRemoteNotificationTypeBadge|UIRemoteNotificationTypeSound|UIRemoteNotificationTypeAlert;
// 注册远程通知(在iOS8被废弃了)
[application registerForRemoteNotificationTypes:remoteTypes];
return YES;
}
- 实现注册远程通知返回deviceToken的回调方法
/**
* 当APNs返回deviceToken会调用,该方法
*
* @param application 当前应用对象
* @param deviceToken 设备令牌
*/
- (void) application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
NSLog(@"%@",deviceToken);
}
- 实现注册远程通知失败回调方法,用于检查失败原因方便调试
/**
* 当从APNs获取deviceToken失败的时候会回调该方法
*
* @param application 应用
* @param error 错误
*/
- (void) application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error
{
NSLog(@"%@",error);
}
- 实现点击远程通知,进入App调用的方法
/**
* 处理点击远程通知的页面跳转处理
*/
- (void) application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
NSLog(@"%s",__FUNCTION__);
}
在Xcode6.4 iOS7的环境下运行程序,得到如下错误
Error Domain=NSCocoaErrorDomain Code=3010 "remote notifications are not supported in the simulator" UserInfo=0x7fc91b467e70 {NSLocalizedDescription=remote notifications are not supported in the simulator}
由于在Xcode6.4下模拟器不能调试推送,DeviceToken都获取不到,我这边没有iOS7的真机设备,所以只能说大致的流程是这样子的.
远程推送通知iOS7与iOS8最主要不同点就是注册远程通知以及iOS8提供了对Catgory的支持,其他的基本都是相同的.
在iOS8以后
- 获取deviceToken
把刚才的程序运行在iOS8.4的真机
他要访问电脑中钥匙串中的证书进行验证,点击"始终允许" 控制台提示
registerForRemoteNotificationTypes: is not supported in iOS 8.0 and later.
也就是说registerForRemoteNotificationTypes:方法在iOS8.0 和以后不再支持了. 一般来说,遇到这样情况,那么在registerForRemoteNotificationTypes方法的注释中,说明在iOS8以后应该怎么做! 我点去这个方法
他说:这个方法在iOS8.0被废弃了,让使用registerForRemoteNotification 和 registerUseNotificationSettings来替代
我要支持iOS7和iOS8需要修改 didFinishLaunchingWithOptions: 方法中注册远程通知的代码,修改后的完整代码,如下:
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// 如果application能够响应这个方法,那么证明当前iOS版本为iOS8.0及以后
if ([application respondsToSelector:@selector(registerUserNotificationSettings:)]) {
// 1.请求用户权限(注册用户通知设置信息),这个方法与本地通知相同
// 1.1 请求通知类型
UIUserNotificationType types = UIUserNotificationTypeAlert|UIUserNotificationTypeBadge|UIUserNotificationTypeSound;
// 1.2 创建用户通知设置信息,categories是用来定制通知的时候的按钮的与本地通知相同
UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:types categories:nil];
// 1.3 注册用户通知设置信息
[application registerUserNotificationSettings:setting];
// 2.注册远程通知
[application registerForRemoteNotifications];
return YES;
}
// 能走到下面说明是iOS8.0之前
// UIRemoteNotificationTypeBadge = 1 << 0, 应用图标右上角数字
// UIRemoteNotificationTypeSound = 1 << 1, 声音
// UIRemoteNotificationTypeAlert = 1 << 2, 提示
// UIRemoteNotificationTypeNewsstandContentAvailability = 1 << 3, 新闻,国内几乎没人使用
// 远程通知类型
UIRemoteNotificationType remoteTypes = UIRemoteNotificationTypeBadge|UIRemoteNotificationTypeSound|UIRemoteNotificationTypeAlert;
// 注册远程通知(在iOS8被废弃了)
[application registerForRemoteNotificationTypes:remoteTypes];
return YES;
}
修改完毕,再次运行程序到真机上,这是用户手机上有提示信息
点击 "好" 即可,如果点击了"不允许"的话,还去设置中进行设置.
等待一分钟左右,会从返回APNs测试服务器返回一个deviceToken,它代表着是那个真机设备.将来Provider发送通知是需要使用到她
<afc70037 3adb7d1c 3529032b be1582b9 49a1a89d 995da320 b6d78c93 b7d9a3a4>
- 使用Provider发送远程通知
- Provider集成是服务端的事情,不用我们搞,我们这里就使用一个Mac版的程序,测试下我们iPhone的程序.
- 步骤:
- 打开 PushMeBaby 项目(在我们当天资料中)
- 把我们生成的SSL开发证书"aps_development.cer"拖入到项目中pushMebaby
- 修改ApplicationDelegate.m的init方法,init方法代码如下
- (id)init {
self = [super init];
if(self != nil) {
self.deviceToken = @"";
self.payload = @"{\"aps\":{\"alert\":\"This is some fancy message.\",\"badge\":1}}";
self.certificate = [[NSBundle mainBundle] pathForResource:@"apns" ofType:@"cer"];
}
return self;
}
- 修改deviceToken为上面获取到deviceToken,注意这个deviceToken 没有前后的尖括号
self.deviceToken = @"afc70037 3adb7d1c 3529032b be1582b9 49a1a89d 995da320 b6d78c93 b7d9a3a4";
- 修改证书名称为我们刚拖入进来证书名称 "aps_development"
self.certificate = [[NSBundle mainBundle] pathForResource:@"aps_development" ofType:@"cer"];
- payload暂且不动,他是我们推送的内容,一会在详细的讲解
- 运行PushMeBaby程序
点击"始终允许",弹出如下窗口
点击 "push" 按钮有可能的错误: SSLWrite(): -9806 0 停止再次运行,点击"push" 按钮一般就好了如果反复几次都是同样的问题,就可以借助一下"百度" 发送成功后,如果是锁屏状态,手机端如下
非锁屏状态下
payload简介
- 首先打开Xcode的文档
- 搜索APNs->推送通知文档
如果想进一步学生推送通知可以看看这个文档,介绍的很详细
- 我们这里来根据文档学习一下payload
这里有payloads的一些例子,payloads是一个json格式字典 "aps"你与系统通知相关的内容,aps意外的都是用户自定的信息当我在代码中获取的userInfo中会显示这些信息我们这里介绍下aps中参数就可以了!
- 拷贝原来pushMeBody中的payloads
{"aps":
{
"alert":"This is some fancy message.",
"badge":1
}
}
alert:表示通知的内容 badge:应用图标右上角红色数字 "sound" : 表示接收通知提示音如:
{"aps":
{
"alert":"This is some fancy message.",
"badge":1,
"sound" :"chime.aif"
}
}
接收通知就会"bingbong"的提示音,提示音还有一个"chime.aiff" 注意:必须保证json的格式是正确的,否则通知发送不出.
接收到远程通知,界面跳转
如果程序没有退出,会执行AppDelegate中的 -[AppDelegate application:didReceiveRemoteNotification:] 方法,在这个方法中可以实现界面跳转,我们这里仅仅是在控制的view上添加一个Label来模拟一下界面跳转,代码如下
/**
* 处理点击远程通知的页面跳转
*/
- (void) application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
UILabel *label = [[UILabel alloc] init];
label.frame = CGRectMake(10, 20, 300, 400);
label.backgroundColor = [UIColor lightGrayColor];
label.numberOfLines = 0;
label.text = [userInfo description];
[self.window.rootViewController.view addSubview:label];
// NSLog(@"%s",__FUNCTION__);
}
运行程序测试,让手机上的测试应用进入后台,注意测试要重新运行下PushMeBady,如果间隔时间比较长的时候,PushMeBady与APNs的链接就断开了,需要重新链接;点击"Push",手机接收到通知消息界面如下:
点通知进入界面,如图
关闭,退出程序手机上的程序,点击PushMeBady上的"Push"按钮,如果你发现没有收到通知,就重新在运行下PushMeBady,然后在点击"Push"按钮,点击通知后,你发现界面上还是白茫茫的一片,
刚才我们不是在接收到的方法,添加了一个Label吗?怎么这里没有呢, 这是因为当关闭应用后,点击通知进入应用不会在调用didReceiveRemoteNotification:方法了,我们想在程序关闭后,跳转界面的话,就需要在didFinishLaunchingWithOptions:方法进行处理在didFinishLaunchingWithOptions:方法的最上面添加如下代码
// 跳转界面
// 如果点击远程通知进来的就跳转界面
// 如果launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]有值就证明是远程通知过来的,他是一个字典
// 取出远程通知内容
NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey];
// 如果有内容就进行跳转
if (userInfo) {
UILabel *label = [[UILabel alloc] init];
label.frame = CGRectMake(10, 20, 300, 400);
label.backgroundColor = [UIColor lightGrayColor];
label.numberOfLines = 0;
label.text = [userInfo description];
[self.window.rootViewController.view addSubview:label];
}
再次运行我们客户端程序->退出应用->运行PushMeBody->点击push按钮->点击通知->界面如下
与didReceiveRemoteNotification接收到内容完全一样
发布含有远程推送服务的应用
如果没有安装发布证书安装下面顺序安装相关证书
- 安装发布证书(ios_distribution.cer)
- 安装远程推送服务SSL证书(aps_proudction.cer)
- 安装Profile(dis_xfypush.mobileprovision)
使用Xcode发布应用
- Product->Archive
- 弹出下面界面
- 选择导出类型
- 选择账号
- 导出
- 结果
使用PushMeBody测试发布状态的远程推送通知的注意点: 开发证书正常。切换到发布证书报错:SSLWrite(): -9806 0 修改 connect 方法中的如下代码
result = SSLSetPeerDomainName(context, "gateway.sandbox.push.apple.com", 30); 改为
result = SSLSetPeerDomainName(context, "gateway.push.apple.com", 22);//22表示gateway.push.apple.com地址的长度