
二、本地推送
1 注册与处理
代码如下:
/// 一般在在启动时注册通知,程序被杀死,点击通知后调用此程序- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0) { // iOS8UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeAlert | UIUserNotificationTypeSound categories:nil];[application registerUserNotificationSettings:setting]; } if (launchOptions[UIApplicationLaunchOptionsLocalNotificationKey]) {// 这里添加处理代码 } return YES;}/// 程序没有被杀死(处于前台或后台),点击通知后会调用此程序- (void)application:(UIApplication *)application didReceiveLocalNotification:(UILocalNotification *)notification { // 这里添加处理代码}可以看到,处理代码有两个方法,一个是- (IBAction)addLocalNotification { // 1.创建一个本地通知 UILocalNotification *localNote = [[UILocalNotification alloc] init]; // 1.1.设置通知发出的时间 localNote.fireDate = [NSDate dateWithTimeIntervalSinceNow:5]; // 1.2.设置通知内容 localNote.alertBody = @"这是一个推送这是一个推送"; // 1.3.设置锁屏时,字体下方显示的一个文字 localNote.alertAction = @"赶紧!!!!!"; localNote.hasAction = YES; // 1.4.设置启动图片(通过通知打开的) localNote.alertLaunchImage = @"../Documents/IMG_0024.jpg"; // 1.5.设置通过到来的声音 localNote.soundName = UILocalNotificationDefaultSoundName; // 1.6.设置应用图标左上角显示的数字 localNote.applicationIconBadgeNumber = 999; // 1.7.设置一些额外的信息 localNote.userInfo = @{@"qq" : @"704711253", @"msg" : @"success"}; // 2.执行通知 [[UIApplication sharedApplication] scheduleLocalNotification:localNote];}效果如下:
3 取消通知
// 取消所有本地通知
[application cancelAllLocalNotifications];
三、远程推送
与Android上我们自己实现的推送服务不一样,Apple对设备的控制非常严格,消息推送的流程必须要经过APNs(Apple Push Notification service).
一般情况下如果一个程序退到后台就不能运行代码(Audio、VoIP等等可以在后台运行),或者程序退出后,那么它就和对应应用的后台服务器断开了链接,就收不到服务器发送的信息了,但是每台设备只要联网就会和苹果的APNs服务器建立一个长连接(persistent IP connection),这样只要通过苹果的APNs服务器,我们自己的服务器就可以间接的和设备保持连接了,示意图如下:

使用步骤:
1 勾选Backgroud Modes -> Remote notifications,主要是iOS7之后,苹果支持后台运行,如果这里打开后,当接收到远程推送后,程序在后台也可以做一些处理,如下图所示:

2 远程推送的注册与本地推送不同,iOS8.0前后也不同,代码见下面。
另外,在第一次使用推送时,可能会有这样的疑问:didFinishLaunchingWithOptions会在每次打开程序时被调用,那是不是每次都会调用注册函数,每次都会弹窗询问用户"是否允许推送通知"?其实这个窗口只会在第一次打开程序时弹出一次,无论用户允许或不允许苹果会记住用户的选择,注册函数调用多次对用户也没什么影响
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // iOS8之后和之前应区别对待 if ([[UIDevice currentDevice].systemVersion floatValue] >= 8.0) {UIUserNotificationSettings *settings = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeBadge | UIUserNotificationTypeAlert | UIUserNotificationTypeSound categories:nil];[[UIApplication sharedApplication] registerUserNotificationSettings:settings]; } else {[application registerForRemoteNotificationTypes:UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIUserNotificationTypeSound]; } return YES;}/// 这个函数存在的意义在于:当用户在设置中关闭了通知时,程序启动时会调用此函数,我们可以获取用户的设置- (void)application:(UIApplication *)application didRegisterUserNotificationSettings:(UIUserNotificationSettings *)notificationSettings { [application registerForRemoteNotifications];}3 如果注册失败,比如没有证书等等,会调用:/// 注册失败调用- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {NSLog(@"远程通知注册失败:%@",error);}4 获取deviceToken/// 用户同意后,会调用此程序,获取系统的deviceToken,应把deviceToken传给服务器保存,此函数会在程序每次启动时调用(前提是用户允许通知)- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken { NSLog(@"deviceToken = %@",deviceToken);}5 用户点击了通知- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler { // userInfo} 注:当在第一步打开后台运行后,用户不点击通知,也可以执行:- (void)application:(UIApplication*)application didReceiveRemoteNotification:(NSDictionary*)userInfo fetchCompletionHandler:(void(^)(UIBackgroundFetchResult))completionHandler5.2 iOS7之前
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { // 获取远程推送消息 NSDictionary *userInfo = launchOptions[UIApplicationLaunchOptionsRemoteNotificationKey]; if (userInfo) {// 有推送的消息,处理推送的消息 } return YES;}/// iOS3之后才有,只有在程序处于后台时,用户点击了通知后才会被调用,应搭配didFinishLaunchingWithOptions使用- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo { // userInfo}在实际编程时,如果想兼容iOS7以前,三个函数可同时使用,都列出来,系统会自动选择合适的调用。{ "aps" : { // 必须有"alert" : "string","body" : "string","badge" : number,"sound" : "string" }, "NotiId" : 20150821,// 自定义key值}8 推送的大小限制