iOS集成极光推送遇到的几个问题
其实官方的 iOS SDK 集成指南和 iOS SDK 教程两个文档关于集成时的大部分问题都可以解决,所以我这里只是讨论我遇见的文档里表述不明白的那几个坑(2.1.0版本的)。
APNs 证书和主证书的区别

我刚开始做推送的时候本来以为 APNs 证书就是主证书里加了个允许推送而已,事实证明我还是太天真了。

首先是创建Provisioning Profile文件时选择证书列表时是不能选择APNs证书的,而且在XCode里Code Signing也是不能选择的。 APNs 证书其实是只表示了有推送功能,主证书还是必不可少的。所以现在只是又添加了两个证书,其他的是不用变的,所以现在就有4个证书了。作为一个有轻微强迫症的人,创建完 APNs 证书就顺手把主证书删了,在这里让我折腾了好久。
启动 SDK 需要传的参数
1
2
3
4+ (void)setupWithOption:(NSDictionary *)launchingOption
appKey:(NSString *)appKey
channel:(NSString *)channel
apsForProduction:(BOOL)isProduction;
前两个就不必说了。 channel 发布渠道。我刚开始也不知道要怎么填,iOS? App Store? Publish channel? 其实这只是一个标识而已,不会有什么具体影响。而且是可选的,所以填nil也是没关系的。 isProduction 是否生产环境。看起来很有用的一个参数,但实际发现 YES 或 NO 是没区别的,在框架里边应该是另有判断。但还必须得填,所以在生产和开发环境之间切换时是没必要一定要改这个参数的。
标签与别名 API
之前的教程文档是没有这个的,下载的Demo里也没有。但是项目里肯定是要有这个的。
别名 alias
为安装了应用程序的用户,取个别名来标识。以后给该用户 Push 消息时,就可以用此别名来指定。每个用户只能指定一个别名。
标签 tag
为安装了应用程序的用户,打上标签。其目的主要是方便开发者根据标签,来批量下发 Push 消息。可为每个用户打多个标签。
1
+ (void) setTags:(NSSet *)tags alias:(NSString *)alias fetchCompletionHandle:(void (^)(int iResCode, NSSet *iTags, NSString *iAlias))completionHandler;
call back 结果可以忽略,但最好判断一下, 返回值为 0 才设置成功,才可以向目标推送。我一般是有 block 就不会用其他的,但使用 block 时需要注意循环引用问题。而且这个方法是覆盖之前的设置,不是新增。
1
2+ (void)setTags:(NSSet *)tags callbackSelector:(SEL)cbSelector object:(id)theTarget;
+ (void)setAlias:(NSString *)alias callbackSelector:(SEL)cbSelector object:(id)theTarget;
tag 和 alias 分别设置会比只调同时设置两个的那个接口稳定,虽然我也不知道为什么。
收到通知后的处理
1
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult result))completionHandler
是在 iOS7 之后的方法,所以之前的方法可以不用了。
1
- (void)applicationWillEnterForeground:(UIApplication *)application
applicationWillEnterForegroun是在didReceiveRemoteNotification之后运行的,所以在这个方法里想处数据是不行的。然后[application setApplicationIconBadgeNumber:0];是偶尔会抽风一下的,所以最好这样写:
1
2[application setApplicationIconBadgeNumber:1];
[application setApplicationIconBadgeNumber:0];
当然,不要忘了[application cancelAllLocalNotifications];
还有,关于跳转到特定页面的问题。反正我是没找到如何在 AppDelegate 里找到当前的页面,所以只能用self.window.rootViewController了。返回的话直接self.view.window.rootViewController = vc;吧。
记得用application.applicationState == UIApplicationStateActive判断一下应用处于的状态。如果用户正在运行应用,或许还有更好的,但我就是弹一个UIAlertView。

