推送通知
1. 什么是推送通知?
> 首先明确:**此处的推送通知跟我们的”NSNotification”没有半毛钱关系
> 可以理解为: 向用户推送一条信息来通知用户某件事情
> 作用: 可以在APP退到后台,或者关闭时;继续推送一条消息告诉用户某件事情
2. 推送通知的应用场景?
(1) 一些任务管理APP,会在任务时间即将到达时,通知你做该任务;
(2) 健身App定时提醒你应该健身了;
(3) 买过电影票后,提前半小时告诉你,电影即将开场;
(4) 当你QQ或者微信收到消息时,即使退到后台,或者关闭APP,也可以收到信息通知告诉我们;
(5) 电商APP,推送一条消息通知我们有新品上架等等
3. 推送通知的展现样式?
(1) 在屏幕顶部显示一块横幅(显示具体内容)
(2) 在屏幕中间弹出一个UIAlertView(显示具体内容)
(3) 在锁屏界面显示一块横幅(锁屏状态下,显示具体内容)
(4) 更新app图标的数字(说明新内容的数量)
(5) 播放音效(提醒作用)
*注意:以上样式只能是用户自己设置,我们无法通过代码控制*
4. 推送通知的分类
-
本地推送通知
“本地”可以理解为”不联网”;即使没有网络情况下,也可以推送通知消息 应用场景: 确定知道未来某个时间点应该提醒用户什么
-
远程推送通知
概念: 与“本地”相对,表示,必须在联网情况下才会向用户推送通知消息 远程推送服务,又称为APNs(Apple Push Notification Services) 应用场景: 1. 不确定未来某个时间点应该提醒用户什么,临时性的 2. 当APP彻底退出时也想继续让用户获取一些最新消息
- 使用原则: 谁能确定通知时间和内容, 谁就可以发送(开发人员在APP内部通过代码发送=本地通知; 服务器可以确定通知时间和内容=远程通知)*
5. 推送通知的实现
1. 本地推送通知
创建UILocalNotification 对象
-
设置一些必要属性
// 推送通知的触发时间(何时发出推送通知) @property(nonatomic,copy) NSDate *fireDate; // 推送通知的具体内容 @property(nonatomic,copy) NSString *alertBody;
-
开始推送通知
// 根据fireDate设定的时间进行推送 [[UIApplication sharedApplication] scheduleLocalNotification:ln]; // 立即推送 presentLocalNotificationNow:(UILocalNotification *)notification;
4. 监听用户点击通知
```objc
// app处于前台
此时不会弹框通知用户通知的到达,但是还是会调用对应的代理方法
- app并没有关闭,一直隐藏在后台时
此时用户点击通知信息后,会让app进入前台,并会调用AppDelegate的下面方法
application: didReceiveLocalNotification:
- app已经被关闭(进程已死)
此时用户点击通知信息后,会启动app,启动完毕会调用AppDelegate的下面方法
application: didFinishLaunchingWithOptions:
launchOptions参数通过UIApplicationLaunchOptionsLocalNotificationKey取出本地推送通知对象
-
额外设置
// 每隔多久重复发一次推送通知 @property(nonatomic) NSCalendarUnit repeatInterval; // 点击推送通知打开app时显示的启动图片 @property(nonatomic,copy) NSString *alertLaunchImage; // 附加的额外信息 @property(nonatomic,copy) NSDictionary *userInfo; // 时区 @property(nonatomic,copy) NSTimeZone *timeZone; (一般设置为[NSTimeZone defaultTimeZone] ,跟随手机的时区) // 在锁屏时显示的动作标题(完整标题:“滑动来” + alertAction) @property(nonatomic,copy) NSString *alertAction; // 音效文件名 @property(nonatomic,copy) NSString *soundName; // app图标数字 @property(nonatomic) NSInteger applicationIconBadgeNumber;
-
其他操作
- 获得被调度(定制)的所有本地推送通知 @property(nonatomic,copy) NSArray *scheduledLocalNotifications; 已经发出且过期的推送通知就算调度结束,会自动从这个数组中移除 - 取消调度本地推送通知 - (void)cancelLocalNotification:(UILocalNotification *)notification; - (void)cancelAllLocalNotifications;
-
注意事项
- 在iOS 8.0+,如果要使用本地通知,需要得到用户的许可 在didFinishLaunchingWithOptions方法中添加如下代码 UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil]; [application registerUserNotificationSettings:settings];
-
额外补充
- 在iOS8.0之后,可以设置推送通知带操作行为 在注册设置时,设置categories:参数 监听通知操作行为的点击
2. 远程推送通知
-
远程推送的原理
> 所有苹果设备, 在联网状态下,都会与苹果服务器建立一个长连接 “长连接”: 相对的一个概念是”短连接” “长连接”优势: 服务器可以向客户端发送信息,保证数据即时性 劣势: 占用客户和服务器资源 “短连接”优势: 节省资源,一个会话结束后,立即释放资源 劣势: 服务器无法主动向客户端发信息 > 苹果设备“长连接”作用: 时间校准 系统升级 查找我的iPhone 推送通知... > 原理就是借助苹果设备与APNs服务器之间的长连接, 借助APNs服务器将消息发送给客户端
-
远程推送通知实现的条件
> 真有真机可以调试推送通知 因为只有真机具备UDID, 才能够生成deviceToken > 需要真机调试证书, 推送测试证书
-
远程推送通知实现, 客户端需要做的事
1. 请求苹果获得deviceToken /** * 请求通知授权,以及请求远程通知 */ - (void)registerRemoteNotification { if (isIOS(8.0)) { // 1. 请求通知授权 UIUserNotificationSettings *notificationSet = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeSound | UIUserNotificationTypeAlert categories:nil]; [[UIApplication sharedApplication] registerUserNotificationSettings:notificationSet]; // 2. 注册远程通知 [[UIApplication sharedApplication] registerForRemoteNotifications]; } else { // iOS8.0之前请求远程推送通知 [[UIApplication sharedApplication] registerForRemoteNotificationTypes:UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound ]; } } 2. 得到苹果返回的deviceToken,发送deviceToken给公司的服务器 /** * 当远程通知注册后,APNs会通过调用这个方法,返回对应的deviceToken */ -(void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken { NSLog(@"%@", deviceToken); } 3. 监听用户对通知的点击 实现APP代理方法 -(void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo { NSLog(@"%@", userInfo); } 或者: /** * 当接收到远程通知时调用(iOS7.0之后使用) * * 当前在前台时; 或者app在后台\app被彻底退出状态下,点击通知打开app进入前台; 都可以执行以下方法 * * * 执行completionHandler 作用 * * 1> 系统会估量App消耗的电量,并根据传递的UIBackgroundFetchResult 参数记录新数据是否可用 * 2> 调用完成的处理代码时,应用的界面缩略图会自动更新 * * 如果想要接收到通知后,不要用户点击通知, 就执行以下代码, 那么必须有三个要求: 1> 必须勾选后台模式Remote Notification ; 2> 告诉系统是否有新的内容更新(执行完成代码块) 3> 设置发送通知的格式("content-available":"随便传") */ -(void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler { }
-
远程推送测试
> PushMeBaby 1. 编译会出现错误, 直接把错误行注释即可; 2. 需要填写对应的deviceToken; 3. 需要将推送测试证书改名为aps.cer 拖入项目中;
-
第三方推送
极光推送(JPush) 个推 信鸽 具体集成步骤以及使用方法, 请查看对应的官方文档, 非常详细; 第三方服务合集: [http://mdsa.51cto.com/services/]
es/]
[http://mdsa.51cto.com/services/]