推送通知是由应用服务提供商发起的,通过苹果的APNs(Apple Push Notification Server)发送到应用客户端。下面是苹果官方关于推送通知的过程示意图:
推送通知的过程可以分为以下几步:
- 应用服务提供商从服务器端把要发送的消息和设备令牌(device token)发送给苹果的消息推送服务器APNs。
- APNs根据设备令牌在已注册的设备(iPhone、iPad、iTouch、mac等)查找对应的设备,将消息发送给相应的设备。
- 客户端设备接将接收到的消息传递给相应的应用程序,应用程序根据用户设置弹出通知消息。
当然,这只是一个简单的流程,有了这个流程我们还无从下手编写程序,将上面的流程细化可以得到如下流程图(图片来自互联网),在这个过程中会也会提到如何在程序中完成这些步骤:
1.应用程序注册APNs推送消息。
说明:
a.只有注册过的应用才有可能接收到消息,程序中通常通过UIApplication的registerUserNotificationSettings:方法注册,iOS8中通知注册的方法发生了改变,如果是iOS7及之前版本的iOS请参考其他代码。
b.注册之前有两个前提条件必须准备好:开发配置文件(provisioning profile,也就是.mobileprovision后缀的文件)的App ID不能使用通配ID必须使用指定APP ID并且生成配置文件中选择Push Notifications服务,一般的开发配置文件无法完成注册;应用程序的Bundle Identifier必须和生成配置文件使用的APP ID完全一致。
2.iOS从APNs接收device token,在应用程序获取device token。
说明:
a.在UIApplication的-(void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken代理方法中获取令牌,此方法发生在注册之后。
b.如果无法正确获得device token可以在UIApplication的-(void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error代理方法中查看详细错误信息,此方法发生在获取device token失败之后。
c.必须真机调试,模拟器无法获取device token。
3.iOS应用将device token发送给应用程序提供商,告诉服务器端当前设备允许接收消息。
说明:
a.device token的生成算法只有Apple掌握,为了确保算法发生变化后仍然能够正常接收服务器端发送的通知,每次应用程序启动都重新获得device token(注意:device token的获取不会造成性能问题,苹果官方已经做过优化)。
b.通常可以创建一个网络连接发送给应用程序提供商的服务器端, 在这个过程中最好将上一次获得的device token存储起来,避免重复发送,一旦发现device token发生了变化最好将原有的device token一块发送给服务器端,服务器端删除原有令牌存储新令牌避免服务器端发送无效消息。
4.应用程序提供商在服务器端根据前面发送过来的device token组织信息发送给APNs。
说明:
a.发送时指定device token和消息内容,并且完全按照苹果官方的消息格式组织消息内容,通常情况下可以借助其他第三方消息推送框架来完成。
5.APNs根据消息中的device token查找已注册的设备推送消息。
说明:
a.正常情况下可以根据device token将消息成功推送到客户端设备中,但是也不排除用户卸载程序的情况,此时推送消息失败,APNs会将这个错误消息通知服务器端以避免资源浪费(服务器端此时可以根据错误删除已经存储的device token,下次不再发送)。
下面将简单演示一下推送通知的简单流程:
首先,看一下iOS客户端代码:
1 // 2 // AppDelegate.m 3 // pushnotification 4 // 5 // Created by Kenshin Cui on 14/03/27. 6 // Copyright (c) 2014年 Kenshin Cui. All rights reserved. 7 // 8 9 #import "AppDelegate.h" 10 #import "KCMainViewController.h" 11 12 @interface AppDelegate () 13 14 @end 15 16 @implementation AppDelegate 17 18 #pragma mark - 应用程序代理方法 19 #pragma mark 应用程序启动之后 20 - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 21 22 _window=[[UIWindow alloc]initWithFrame:[UIScreen mainScreen].bounds]; 23 24 _window.backgroundColor =[UIColor colorWithRed:249/255.0 green:249/255.0 blue:249/255.0 alpha:1]; 25 26 //设置全局导航条风格和颜色 27 [[UINavigationBar appearance] setBarTintColor:[UIColor colorWithRed:23/255.0 green:180/255.0 blue:237/255.0 alpha:1]]; 28 [[UINavigationBar appearance] setBarStyle:UIBarStyleBlack]; 29 30 KCMainViewController *mainController=[[KCMainViewController alloc]init]; 31 _window.rootViewController=mainController; 32 33 [_window makeKeyAndVisible]; 34 35 //注册推送通知(注意iOS8注册方法发生了变化) 36 [application registerUserNotificationSettings:[UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeAlert|UIUserNotificationTypeBadge|UIUserNotificationTypeSound categories:nil]]; 37 [application registerForRemoteNotifications]; 38 39 return YES; 40 } 41 #pragma mark 注册推送通知之后 42 //在此接收设备令牌 43 -(void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken{ 44 [self addDeviceToken:deviceToken]; 45 NSLog(@"device token:%@",deviceToken); 46 } 47 48 #pragma mark 获取device token失败后 49 -(void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error{ 50 NSLog(@"didFailToRegisterForRemoteNotificationsWithError:%@",error.localizedDescription); 51 [self addDeviceToken:nil]; 52 } 53 54 #pragma mark 接收到推送通知之后 55 -(void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo{ 56 NSLog(@"receiveRemoteNotification,userInfo is %@",userInfo); 57 } 58 59 #pragma mark - 私有方法 60 /** 61 * 添加设备令牌到服务器端 62 * 63 * @param deviceToken 设备令牌 64 */ 65 -(void)addDeviceToken:(NSData *)deviceToken{ 66 NSString *key=@"DeviceToken"; 67 NSData *oldToken= [[NSUserDefaults standardUserDefaults]objectForKey:key]; 68 //如果偏好设置中的已存储设备令牌和新获取的令牌不同则存储新令牌并且发送给服务器端 69 if (![oldToken isEqualToData:deviceToken]) { 70 [[NSUserDefaults standardUserDefaults] setObject:deviceToken forKey:key]; 71 [self sendDeviceTokenWidthOldDeviceToken:oldToken newDeviceToken:deviceToken]; 72 } 73 } 74 75 -(void)sendDeviceTokenWidthOldDeviceToken:(NSData *)oldToken newDeviceToken:(NSData *)newToken{ 76 //注意一定确保真机可以正常访问下面的地址 77 NSString *urlStr=@"http://192.168.1.101/RegisterDeviceToken.aspx"; 78 urlStr=[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; 79 NSURL *url=[NSURL URLWithString:urlStr]; 80 NSMutableURLRequest *requestM=[NSMutableURLRequest requestWithURL:url cachePolicy:0 timeoutInterval:10.0]; 81 [requestM setHTTPMethod:@"POST"]; 82 NSString *bodyStr=[NSString stringWithFormat:@"oldToken=%@&newToken=%@",oldToken,newToken]; 83 NSData *body=[bodyStr dataUsingEncoding:NSUTF8StringEncoding]; 84 [requestM setHTTPBody:body]; 85 NSURLSession *session=[NSURLSession sharedSession]; 86 NSURLSessionDataTask *dataTask= [session dataTaskWithRequest:requestM completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { 87 if (error) { 88 NSLog(@"Send failure,error is :%@",error.localizedDescription); 89 }else{ 90 NSLog(@"Send Success!"); 91 } 92 93 }]; 94 [dataTask resume]; 95 } 96 @end
iOS客户端代码的代码比较简单,注册推送通知,获取device token存储到偏好设置中,并且如果新获取的device token不同于偏好设置中存储的数据则发送给服务器端,更新服务器端device token列表。