美文网首页iOS技术IOS网友们的篇章iOS开发超神学院
04.iOS远程推送通知 APNs远程推送,极光推送

04.iOS远程推送通知 APNs远程推送,极光推送

作者: Liwx | 来源:发表于2016-01-16 01:37 被阅读3641次

    @(〓〓 iOS-实用技术)[远程/本地推送]


    目录

    • 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生成证书
    01.真机调试证书和推送通知调试证书.png

    远程推送的原理

    • 所有苹果设备, 在联网状态下,都会与苹果服务器建立一个长连接

      “长连接”: 相对的一个概念是”短连接”
      - “长连接”
      优势: 服务器可以向客户端发送信息,保证数据即时性
      劣势: 占用客户端和服务器资源
      - “短连接”
      优势: 节省资源,一个会话结束后,立即释放资源
      劣势: 服务器无法主动向客户端发信息

    • 苹果设备“长连接”作用:

      • 时间校准
      • 系统升级
      • 查找我的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项目中;
    02.真机测试证书改名为aps.cer.png
    • 编译成功后,运行效果图
    03.运行效果图.png
    • PushMeBaby设置deviceToken的两种方式
    04.PushMeBaby设置deviceToken的两种方式.png

    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 回调代码块
    // ----------------------------------------------------------------------------
    // 监听远程推送通知点击(优先级较高),一接收到通知就会立即调用该方法
    /**
        当接收到推送通知之后, 并且满足一下条件
        当我们实现, 这个方法时, 上面一个方法不再执行
        计时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
    05.真机调试证书,推送调试证书和描述文件profile.png

    安装证书和描述文件后的效果图

    06.安装证书和描述文件后的效果图.png
    • 2.配置Bundle ID必须要和开发者中心配置的Bundle ID一致
    07.配置Bundle ID.png
    • 3.设置开启应用的后台模式(要实现一接收到通知就会立即调用的功能必须配置后台模式)
    08.设置开启应用的后台模式.png
    • 4.配置开发者真机调试
      • 选择Build Settings,搜索code signing
    09.配置开发者真机调试.png 10.配置开发者真机调试.png

    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(两种方式任选一种)

    11.需要填写iPhone客户端向苹果请求对应的deviceToken.png
    • 3.如果要让iOS应用接收到通知立即执行响应操作

      • 必须在PushMeBaby中将payload数据内容添加"content-available":"随便传"
    • 需要将开发者中心生成的推送真机测试证书改名为aps.cer 拖入PushMeBaby项目中参与编译;

    12.推送真机测试证书改名为aps.cer 拖入PushMeBaby项目.png

    运行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 推送更简单,也另外提供应用内消息推送。这在类似于聊天的场景里很有必要。


    极光推送介绍

    • 极光推送发送通知流程图
    13.极光推送发送通知流程图.png

    极光推送的集成

    • JPush iOS SDK 集成指南

    链接: http://docs.jpush.io/guideline/ios_guide/

    • JPush iOS SDK 教程

    链接: http://docs.jpush.io/client/ios_tutorials/

    • JPush SDK下载

    链接: https://www.jpush.cn/common/products

    相关文章

      网友评论

      • 心语风尚:'CoreServices/../Frameworks/CarbonCore.framework/Headers/MacTypes.h' file not found 为什么
      • bb3b43eb7907:你好,我把插件注入官微,重签就不推送了。请问一下极光这个怎么才能推送
      • Gary_fei:push之后报“you need the APNS Certificate for the app to work”,其中apps.cer推送开发正式已经添加过了
      • 话少为环保:为什么要导入推送证书呢?
      • zero000:如果aps不包含"content-available"字段,那么在程序退出的状态下,点击通知会调用收到通知的fetch方法吗?
      • Sumency:到底是aps.cer还是apns.cer啊
        hd_hd:这个不重要,只要代码对应好就行了
      • Aiverson:不错,很系统

      本文标题:04.iOS远程推送通知 APNs远程推送,极光推送

      本文链接:https://www.haomeiwen.com/subject/dqphkttx.html