iOS 推送使用自有服务踩坑记

在正常的项目开发中,使用第三方友盟、极光等推送的场景居多,这些三方平台对接的其实还是苹果自身的推送服务,在这些服务的基础上做了一些特殊的优化,并且整合了安卓平台。
如果是国外应用,或者客户能使用Google play 服务,推荐使用firebase 集成,这个平台不仅支持安卓和iOS,flutter端也有官方的plugin,不用自己去写桥接方法了。
下面从几个方面介绍推送服务搭建过程中的一些坑点。

1、推送证书

推送证书iOS分为两类,一种是 P12 证书、一种 P8 证书。
P12 证书是与环境、应用深度绑定的证书,需要与应用的bundleId配合使用??梢杂辛教?,一套沙盒环境(测试环境),一套正式环境,在后端对应的服务地址是不一样。证书需要每年更新一次,会有过期时间。
P8 证书是苹果最近这几年优化后的证书,每个账户可以申请一个,申请后账号下所用应用都可以使用,不需要为新增的应用额外添加推送证书,并且长期有效,不需要每年更新,这种方式确实对开发及管理者都是很友好的。
推荐使用P8,这种后期维护的成本基本没有。

2、DeviceToken

1·token绑定

这边有个小坑,DeviceToken原生获取的时候只有用户允许了通知权限才会触发DeviceToken获取的代理方法,所以刚进入应用的时候可能拿不到DeviceToken,但是我们的一般接口设计的时候可能将DeviceToken与登录接口关联起来,那这边就会有个小bug,万一用户第一次安装App并且打开App的时候没有允许通知,然后登陆了App后又想起来要允许通知,又跑去设置——通知里去把通知权限打开,那么这里就会错过一个用户绑定DeviceToken的机会,所以一般需要一个单独接口来更新token,当然可以将登录接口那边的token绑定逻辑去掉,不去其实问题也不大,反正绑定肯定是需要用户登录完成后才能进行的。
这里你可能会问为啥第三方的推送没这个问题,这里以最常用的极光推送来举例,极光提供一个获取RegisterID的方法,但这个不是DeviceToken,DeviceToken是需要调用苹果自有代理获取的,这两个不是一个东西,极光貌似还有个tagID,那个没有用过,好像是标签ID,是用来给用户分组,便于给特定标签的用户推送的。在使用极光推送的时候,自有后台对接的主要是RegisterID,而App将DeviceToken利用极光的App SDK 丢给极光,在极光推送管理平台上也是用RegisterID模拟推送,所以给我们的假象就是极光用RegisterID在推送。笔者这边大胆猜测一下,其实极光RegisterID可是极光生成的设备与应用的唯一码,一旦获取到DeviceToken,再将RegisterID和DeviceToke绑定起来,就可以完美躲避DeviceToke获取的时机,这只是我的假设。
所以,DeviceToken的绑定只放在登录接口是不完美的。
如何是允许多设备登录,那维护的成本貌似有点高,还好做即时通讯基本都是单点登录。

2·token获取

DeviceToken貌似返回有两种格式,具体的好像要看系统版本,这个网上能百度到,这种应该貌似是13以上的获取方式

#include <arpa/inet.h>
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
    if (![deviceToken isKindOfClass:[NSData class]]) return;
    const unsigned *tokenBytes = (const unsigned *)[deviceToken bytes];
    NSString *hexToken = [NSString stringWithFormat:@"%08x%08x%08x%08x%08x%08x%08x%08x",
                          ntohl(tokenBytes[0]), ntohl(tokenBytes[1]), ntohl(tokenBytes[2]),
                          ntohl(tokenBytes[3]), ntohl(tokenBytes[4]), ntohl(tokenBytes[5]),
                          ntohl(tokenBytes[6]), ntohl(tokenBytes[7])];
    NSLog(@"deviceToken:%@",hexToken);
}

还有一种是比较常用的

- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
    NSString *token = [[deviceToken description] stringByTrimmingCharactersInSet:[NSCharacterSet characterSetWithCharactersInString:@"<>"]];
    NSLog(@"deviceToken=%@", deviceToken);
}
3·token解绑

清token的逻辑一般放在退出登录接口。
若是App 是https业务接口,一般App都有一些异常直接跳转登录页的情况,那这个时候用户可能会登录一个不是之前登录的账号,那这样等于一个token绑了两个用户,发推送就会有些异常情况。
如果要彻底点,直接绑定的时候看有没有其他用户绑这个token,如果其他用户绑的要把其他绑定的解绑了再和当前用户进行绑定。

3、推送发送时机

正常https貌似没什么需要注意的,有需要就可以发。
如果是websocket的话可能需要注意一点,不知道微信QQ这类如何设计的,反正我们这边设计是将调用推送接口的机会降到最低了。
1、心跳包收不到,相当于用户离线,这时候有消息需要发推送
2、增加App进入后台和回到前台状态的信令或接口高数后端服务,只在进入后台收到消息的情况下发送推送

4、角标

iOS这边一直是角标和推送是一套内容体系,安卓的貌似是两回事,每次和安卓沟通的时候安卓都说不是一个需求,实在不太懂安卓。
角标这个其实极光做的很不错的,它内部应该是一套角标逻辑体系的,居然能耦合+1的这种设置法,在iOS原生推送里,badge就只能是integer数值,这种+1的方式大大减轻了开发的维护成本。
但现在要抛开三方,自己实现的话,恐怕后台就必须维护这个角标数值了。
大体逻辑是这样:
1、服务端将角标数与用户关联,发送推送的时候需要将角标数+1,设置到推送内容里,并且更新到表内,这样下次推送和这个逻辑一样
2、服务端需要给前端提供一个可以更新角标数的接口,让App更新角标数。
这样就能保证App业务上的未读数量和下次推送角标数的准确性。
你可能能会问即时通信那边怎么做的,即时通信应用是有消息缓存的,前后台逻辑和上面的大体一致,只是App端多了一个汇总所有未读消息数的逻辑,在App退至后台可以将未读数告诉后台。

5、推送内容

在即时通信应用里,信令是很重要的东西,所以推送的扩展内容会把信令带进去,这样,用户点击推送的时候就知道该处理什么样的逻辑了。
比如音视频通话的推送,虽然用户刚登陆账号,可以从离线消息里面拿到音视频的消息,但是离线消息只是为了会话列表展示数据所用,真正需要进行的通话还是要从推送里去获取,可能来了四五个通话,而用户真正想处理的是第一个通话,而不是最后发起的那个。
通过扩展内容,可以拿到建立通话的基本信息。
所以,一切业务需要的不需要展示的信息都可以放到拓展字段里。

结语

上面是针对项目的一些总结,如果写的不对的或疑问的地方欢迎大家批评指正,或者有需要补充的内容也欢迎大家私信我。

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容

  • 原创:知识点总结性文章创作不易,请珍惜,之后会持续更新,不断完善个人比较喜欢做笔记和写总结,毕竟好记性不如烂笔头哈...
    时光啊混蛋_97boy阅读 1,362评论 0 7
  • 前言 最近公司要求把维护期的项目都集成极光推送,集成期间遇到一些小坑,特此在这总结! 极光推送能干嘛? 1.为 J...
    Tate_code阅读 22,630评论 46 70
  • 前言 现在第三方推送也很多 ,比如极光,融云,信鸽,其原理也是相同利用APNS推送机制 ,公司让做自己的推送。避免...
    修_远阅读 10,831评论 9 19
  • 视频https://community.jiguang.cn/t/jpush-ios-sdk/4247 1.iOS...
    An_Jun阅读 11,328评论 0 1
  • 极光推送: 1.JPush当前版本是1.8.2,其SDK的开发除了正常的功能完善和扩展外也紧随苹果官方的步伐,SD...
    Isspace阅读 6,713评论 10 16