问题背景:
项目中发现有一台手机收不到推送消息,后台信息显示RegistrationID为空。
相关知识点:
RegistrationID 的定义
关于 RegistrationID 极光官方文档有如下的定义:
集成了 JPush SDK 的应用程序在第一次 App 启动后,成功注册到 JPush 服务器时,JPush 服务器会给客户端返回唯一的该设备的标识 – RegistrationID。JPush SDK 会以广播的形式发送 RegistrationID 到应用程序。
有了这个标识,App 编程可以把这个 RegistrationID 保存到自己的应用服务器上,然后就可以根据 RegistrationID 来向设备推送消息或者通知。
RegistrationID 变化可能性
1.如果 App 不卸载,是直接覆盖安装, iOS 上 RegistrationID 的值都不会变化。
2.如果 App 是卸载之后再次安装:iOS 上如果启用了 IDFA 变化可能性不大,如果未启用 IDFA 则每次安装 RegistrationID 都会变;
RegistrationID 生成规则解析
鉴于 iOS 系统设计上限制设备唯一标识,所以极光一直使用 Device Token 作为标识,也因为极光推送本身就是需要 Device Token 这个值才可能运作的。
iOS 9 版本之后,每次卸载后重装都会导致 Device Token 变化,所以对于极光后台来说,都只能被识别为新用户。
极光 SDK 新版本增加了 IDFA 选项,在集成初始化 SDK 时可选把 IDFA 这个值设置进来,这样极光后台就优先根据 IDFA 值来识别用户,从有一定的可能性应用被卸载后重装还能识别回老设备。
IDFA 是广告标识符,是 iOS 专门为广告跟踪唯一地识别用户而设计的。在 iOS 设备上,设备 -> 隐私 -> 广告这个页面,有一个设置项:限制广告跟踪。默认是未选中状态的,即是关闭状态,是不限制的。用户可以选中,从而限制广告跟踪。设置项之外还有一个按钮:还原广告标识符…。如果用户点击了这个按钮,则 IDFA 值会变化。
默认的情况下,没有限制广告跟踪,可以取到 IDFA 这个值。并且用户未点击『还原广告标识』时,这个值是不会变的。这样就达到了唯一地标识设备、跟踪到用户的目标。
总之: Device Token + IDFA = RegistrationID;
解决方案:
因为RegistrationID 是在JPush服务器注册后才得到的,所以在调用SDK获取 RegistrationID时,并不能马上得到该值。可以在ios上监听kJPFNetworkDidLoginNotification
通知的代码里获取该值。
代码如下
- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions {
//进行推送消息的注册
[APServiceregisterForRemoteNotificationTypes:(UIUserNotificationTypeBadge|UIUserNotificationTypeSound|UIUserNotificationTypeAlert) categories:nil];
[APServicesetupWithOption:launchOptions];
NSNotificationCenter*defaultCenter =[NSNotificationCenterdefaultCenter];
[defaultCenteraddObserver:self
selector:@selector(networkDidLogin:)
name:kJPFNetworkDidLoginNotification
object:nil];
returnYES;
}
- (void)networkDidLogin:(NSNotification*)notification {
NSLog(@"已登录");
NSString *registrationIDStr = [APService registrationID];
if(![CommonUtility isBlankString:registrationIDStr]){
[UserInfo saveRegisterId:registrationIDStr];
}
}