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的真机设备,所以只能说大致的流程是这样子的.

远程推送通知iOS7iOS8最主要不同点就是注册远程通知以及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来替代

我要支持iOS7iOS8需要修改 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程序

点击"始终允许",弹出如下窗口 

09DC35E4-46D1-4C26-89BE-596AAB10670A.png.jpg

点击 "push" 按钮有可能的错误: SSLWrite(): -9806 0 停止再次运行,点击"push" 按钮一般就好了如果反复几次都是同样的问题,就可以借助一下"百度" 发送成功后,如果是锁屏状态,手机端如下 

1EE65754-BD06-421F-9AB3-E52CB8121095.png.jpg

非锁屏状态下

E8AA3249-4477-4539-B62A-9831F1B667E1.png.jpg

payload简介

  • 首先打开Xcode的文档
  • 搜索APNs->推送通知文档

B7B644DE-95D0-4443-815B-94736FBADE3F.png.jpg

 如果想进一步学生推送通知可以看看这个文档,介绍的很详细

  • 我们这里来根据文档学习一下payload 

DE3BFE9E-EEFA-4736-900C-51344BBCF67E.png.jpg

这里有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,如果间隔时间比较长的时候,PushMeBadyAPNs的链接就断开了,需要重新链接;点击"Push",手机接收到通知消息界面如下

E8AA3249-4477-4539-B62A-9831F1B667E1.png.jpg

 点通知进入界面,如图

FEE1FF8A-F9B4-4B16-9B5A-FEDF2EB87E13.png.jpg

关闭,退出程序手机上的程序,点击PushMeBady上的"Push"按钮,如果你发现没有收到通知,就重新在运行下PushMeBady,然后在点击"Push"按钮,点击通知后,你发现界面上还是白茫茫的一片,

C961D335-40E6-4DA6-B72F-EBA0D95BB2D6.png.jpg

刚才我们不是在接收到的方法,添加了一个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按钮->点击通知->界面如下

FEE1FF8A-F9B4-4B16-9B5A-FEDF2EB87E13.png.jpg

didReceiveRemoteNotification接收到内容完全一样

发布含有远程推送服务的应用

如果没有安装发布证书安装下面顺序安装相关证书

  1. 安装发布证书(ios_distribution.cer)
  2. 安装远程推送服务SSL证书(aps_proudction.cer)
  3. 安装Profile(dis_xfypush.mobileprovision)

使用Xcode发布应用

  1. Product->Archive 

DD47C7E8-5C88-4BDB-A3ED-240178D07126.png.jpg

  1. 弹出下面界面 

3F6B8D13-27E0-4DE8-9118-63C0AD756FCE.png.jpg

  1. 选择导出类型

C4830795-13F7-400D-8123-1CA9B2B8652C.png.jpg

  1. 选择账号

4D73CFE7-7909-4D3F-96A3-F2CDEC9F4153.png.jpg

  1. 导出

5DB8C76F-6127-4045-8D28-802E6DC953AF.png.jpg

  1. 结果

5A77DB5A-B0B3-4C1D-82AB-3EDB3681D75B.png.jpg

使用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地址的长度

原文地址:https://www.cnblogs.com/SingCnblogs/p/5128968.html