前言
本文主要讲本地通知和远程通知使用的一些注意点。
1. 不同系统注册远程通知的方法
注册远程通知的方法:在iOS8以上系统使用如下方法注册,注意点如果是注册本地通知也需要写如下代码才可以收到本地通知。因为本地通知和远程通知的UI界面都是一致的。
// 注册接收通知的类型
UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeAlert | UIUserNotificationTypeBadge | UIUserNotificationTypeSound categories:nil];
[application registerUserNotificationSettings:settings];
// 注册允许接收远程推送通知
[application registerForRemoteNotifications];
// 在iOS7已下系统使用如下方法注册远程通知
[application registerForRemoteNotificationTypes:UIUserNotificationTypeAlert | UIUserNotificationTypeBadge | UIUserNotificationTypeSound];
2. 在application相应的代理方法中处理远程通知
// 当得到苹果的APNs服务器返回的DeviceToken就会被调用
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
NSLog(@"deviceToken是:%@", deviceToken);
}
// 接收到远程通知,触发方法
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
NSLog(@"%@", userInfo);
}
3. 使用后台的远程消息推送
1. 首先在Capabilities中打开远程推送通知 开关
2. 实现代理方法:
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
3. 远程消息数据格式:
{"aps" : {"content-available" : 1},"content-id" : 42}
4. 执行completionHandler有两个目的
- 系统会估量App消耗的电量,并根据传递的UIBackgroundFetchResult 参数记录新数据是否可用
- 调用完成的处理代码时,应用的界面缩略图会自动更新
注意点:接收到远程通知到执行完网络请求之间的时间不能超过30秒(后台刷新 ?)
if (userInfo) {
int contentId = [userInfo[@"content-id"] intValue];
ViewController *vc = (ViewController *)application.keyWindow.rootViewController;
[vc loadDataWithContentID:contentId completion:^(NSArray *dataList) {
vc.dataList = dataList;
NSLog(@"刷新数据结束");
completionHandler(UIBackgroundFetchResultNewData);
}];
} else {
completionHandler(UIBackgroundFetchResultNoData);
}
3. 本地通知的代理方法
3.1 接收到本地通知时会调用这个方法:情况如下
- 如果应用在后台,通过点击通知来打开应用会来到该代理方法
- 如果应用在前台,接受到本地通知就会调用该方法
- 如果应用被杀死后,通过点击通知来打开应用,此时不会来到该代理方法
- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification
需要针对2.3点进行相应的处理(远程通知同样如此,只是代理方法不同而已):
- 判断如果app当前是在前台接收到本地通知,也会调用这个代理方法,此时需要进行过滤。
// UIApplicationStateInactive:从后台到前台的过程
if (application.applicationState == UIApplicationStateActive) return;
if (application.applicationState == UIApplicationStateInactive) {
NSLog(@"跳转相应的界面");
}
- 如果程序已退出(杀死掉了),此时收到本地通知,如过点击本地通知启动app,它就不会再调用这个代理方法了。如果需要跳转相应界面,需要另行处理。可以在下面这个方法中进行跳转。
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;
- 如果应用是正常打开的,launchOptions就是null,如果是非正常打开的,launchOptions就有值。什么是非正常打开:例如通过远程推送打开应用、通过别的应用打开本应用、通过本地通知打开本应用
// 本地通知的情况
if (launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]) {
NSLog(@"跳转相应的界面");
}
// 远程通知的情况
如果launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]有值,说明是应用程序是通过点击远程通知来启动的
if (launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]) {
// 实现跳转的代码。。。
}