一.關於推送通知
來源:http://blog.csdn.net/enuola/article/details/8627283
推送通知,也被叫做遠程通知,是在iOS 3.0以后被引入的功能。是當程序沒有啟動或不在前台運行時,告訴用戶有新消息的一種途徑,是從外部服務器發送到應用程序上的。一般說來,當要顯示消息或 下載數據的時候,通知是由遠程服務器(程序的提供者)發送,然后通過蘋果的推送通知服務(Apple Push Notification Service,簡稱apns)推送到設備的程序上。
推送的新消息可能是一條信息、一項即將到期的日程或是一份遠程服務器上的新數據。在系統上展現的時候,可以顯示警告信息或在程序icon上顯示數字,同 時,也可以播放警告音。一旦用戶注意到程序有新的信息、時間或是數據,他們可以運行程序並訪問新的內容。也可以選擇忽略通知,這時程序將不會被激活。
iPhone, iPad和iPod touch上同一時刻只有一個app在前台運行。大多數程序在后台運行的時候,可以對某些用戶感興趣的內容做出回應(定時、或數據等)。推送通知能讓程序在這些事件發生的時候通知用戶。
作為提供者為程序開發和部署推送通知,必須通過iOS Developer Program Portal獲得SSL證書。每個證書限用於一個程序,使用程序的bundle ID作為標識。證書有兩種用途的:一種是針對sandbox(用於開發和測試),另外一種針對發布產品。這兩種運行環境擁有為各自指定的IP地址並且需要 不同的證書。還必須為兩種不同的環境獲取各自的provisioning profiles。
APNS提供了兩項基本的服務:消息推送和反饋服務。
消息推送:使用流式TCP套接字將推送通知作為二進制數據發送給APNs。消息推送有分別針對開發和測試用的sandbox、發布產品的兩個接口,每個都 有各自的地址和端口。不管用哪個接口,都需要通過TLS或SSL,使用SSL證書來建立一個安全的信道。提供者編制通知信息,然后通過這個信道將其發送給 APNs。
注:sandbox: gateway.sandbox.push.apple.com:219
產品接口:gateway.push.apple.com:2195
反饋服務:可以得到針對某個程序的發送失敗記錄。提供者應該使用反饋服務周期性檢查哪些設備一直收不到通知,不需要重復發送通知到這些設備,降低推送服務器的負擔。
注:sandbox:feedback.push.apple.com:2196
產品接口:feedback.sandbox.push.apple.com:2196
二.Apple Push Notification的工作機制
下面是一個完整推送流程圖
從上圖,我們可以看到。
- 首先是應用程序注冊消息推送。
- IOS跟APNS Server要deviceToken。應用程序接受deviceToken。
- 應用程序將deviceToken發送給PUSH服務端程序(Provider)。
- 服務端程序向APNS服務發送消息。
- APNS服務將消息發送給iPhone應用程序。
無論是iPhone客戶端跟APNS,還是Provider和APNS都需要通過證書進行連接的:
圖中,
1. Provider是指某個iPhone軟件的Push服務器,是我們將要開發的服務器。
2. APNS 是Apple Push Notification Service(Apple Push服務器)的縮寫,是蘋果的服務器。
上圖可以分為三個階段:
第一階段:推送服務器(provider)把要發送的消息、目的iPhone的標識打包,發給APNS;
第二階段:APNS在自身的已注冊Push服務的iPhone列表中,查找有相應標識的iPhone,並把消息發到iPhone;
第三階段:iPhone把發來的消息傳遞給相應的應用程序,並且按照設定彈出Push通知。
三.開發證書和推送證書的配置
1. 使用開發者帳號登錄IOS Provisioning ,選擇或新建一個App Id,這里以“info.luoyl.iostest”為例
2. 創建完后,進入App Id列表,可以看到新建的App Id默認是沒有激活推送功能的,點擊Configure鏈接,進入推送功能激活頁面:
3. 在“Enable for Apple Push Notification service”選項上打勾,然后在行點“configure”按鈕:
4. 此時會彈出一窗口,點“continue”
5. 彈出證書上傳頁面,證書選擇事先做好的“CertificateSigningRequest.certSigningRequest”,然后點“Generate”按鈕;
6. 接下來會有“Your APNs SSL Certificate has been generated.”提示,點“continue”:
7. 下載剛生成的證書“aps_development.cer”到電腦:
8. 至此,appid的Development Push SSL Certificate已經變成“Enabled”狀態了:
9. 制作一開發者測試證書,appid指定為“info.luoyl.iostest”, 下載后雙擊安裝到電腦上:
10. 雙擊在步驟7下載的“aps_development.cer”安裝到keychain Access上:
11. 選中push Services證書,右鍵導出證書為個人信息交換(.p12)格式文件,這里我命名為“aps_development.p12”,點存儲時會彈出一個密碼設置窗口,可留空不填:
12. 在終端執行下面的命令,把剛才導出的個人信息交換(.p12)格式文件加密轉換成推送服務器的推送證書:
- openssl pkcs12 -clcerts -nokeys -out cert.pem -in aps_development.p12
- openssl pkcs12 -nocerts -out key.pem -in aps_development.p12
- openssl rsa -in key.pem -out key.unencrypted.pem
- cat cert.pem key.unencrypted.pem > iostest_push_dev.pem
上面的命令在執行時有4處是需要輸入密碼的,其中1和2直接回車,3必須設定一個key如“push”,在4處輸入3設定的key “push”;
命令執行完后生成的“iostest_push_dev.pem”就是我們推送服務器要使用的推送證書;
經過以上步驟的配置,已經完成了開發推送功能所需要的條件了,接下來將會新建一個ios應用來體驗完成推送功能,在ios應用需要實現的接口。
四.開發帶有推送功能的IOS應用
為使應用能支持推送功能,我們的項目配置時要注意:
- Bundle Identifier、Code Signing指定的開發證書綁定的AppId要和推送證書綁定的AppId一致(見下圖);
- 如 果項目中的開發證書在AppId激活推送功能前已經創建了,這時必須重新生成一個。支持推送功能的開發證書會比舊證書多出一項名為 “aps- environment”的授權串,如果繼續使用舊證書,在程序啟動嘗試注冊推送功能時會出現“ 未找到應用程序的“aps-environment”的權利字符串 ”的錯誤;
- 測試需要用真機,模擬器不支持。
在代碼方面,推送的注冊、監聽和處理都集中在AppDelegate類里:
1.(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
在該方法體里主要實現兩個功能:
一是完成推送功能的注冊請求,即在程序啟動時彈出是否使用推送功能;
二是實現的程序啟動是通過推送消息窗口觸發的,在這里可以處理推送內容;
- - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
- {
- self.window = [[[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]] autorelease];
- // Override point for customization after application launch.
- self.viewController = [[[ViewController alloc] init] autorelease];
- self.window.rootViewController = self.viewController;
- [self.window setBackgroundColor:[UIColor colorWithPatternImage:[UIImage imageNamed:@"background.png"]]];
- [self.window makeKeyAndVisible];
- /** 注冊推送通知功能, */
- [[UIApplication sharedApplication] registerForRemoteNotificationTypes:(UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound)];
- //判斷程序是不是由推送服務完成的
- if (launchOptions) {
- NSDictionary* pushNotificationKey = [launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
- if (pushNotificationKey) {
- UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"推送通知"
- message:@"這是通過推送窗口啟動的程序,你可以在這里處理推送內容"
- delegate:nil
- cancelButtonTitle:@"知道了"
- otherButtonTitles:nil, nil];
- [alert show];
- [alert release];
- }
- }
- return YES;
- }
2. 接收從蘋果服務器返回的唯一的設備token,該token是推送服務器發送推送消息的依據,所以需要發送回推送服務器保存
- - (void)application:(UIApplication *)app didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
- NSString* token = [NSString stringWithFormat:@"%@",deviceToken];
- NSLog(@"apns -> 生成的devToken:%@", token);
- //把deviceToken發送到我們的推送服務器
- DeviceSender* sender = [[[DeviceSender alloc]initWithDelegate:self ]autorelease];
- [sender sendDeviceToPushServer:token ];
- }
3.接收注冊推送通知功能時出現的錯誤,並做相關處理:
- - (void)application:(UIApplication *)app didFailToRegisterForRemoteNotificationsWithError:(NSError *)err {
- NSLog(@"apns -> 注冊推送功能時發生錯誤, 錯誤信息:\n %@", err);
- }
4. 接收到推送消息,解析處理
- - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
- {
- NSLog(@"\napns -> didReceiveRemoteNotification,Receive Data:\n%@", userInfo);
- //把icon上的標記數字設置為0,
- application.applicationIconBadgeNumber = 0;
- if ([[userInfo objectForKey:@"aps"] objectForKey:@"alert"]!=NULL) {
- UIAlertView* alert = [[UIAlertView alloc] initWithTitle:@"**推送消息**"
- message:[[userInfo objectForKey:@"aps"] objectForKey:@"alert"]
- delegate:self
- cancelButtonTitle:@"關閉"
- otherButtonTitles:@"處理推送內容",nil];
- alert.tag = alert_tag_push;
- [alert show];
- }
- }
通過上面的代碼,基本推送功能的開發已經完成了。最后附件上面代碼中所需用到的DeviceSender的類文件,需要將其頭文件導入到AppDelegate中。下面是DeviceSender類的.h和.m文件的下載地址: