iOS開發如何實現消息推送機制


  (2012-12-19 14:22:54)


原文http://www.iwebmobile.com/    html

相關文章:http://blog.csdn.net/r8hzgemq/article/details/7874914 ios

一.關於推送通知 web

推送通知,也被叫作遠程通知,是在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推送通知 push 01 iOS開發:推送通知簡述及開發實踐 (重要!)

 

從上圖,咱們能夠看到。 app

  1.   首先是應用程序註冊消息推送。
  2.  IOS跟APNS Server要deviceToken。應用程序接受deviceToken。
  3.  應用程序將deviceToken發送給PUSH服務端程序(Provider)。
  4.  服務端程序向APNS服務發送消息。
  5.  APNS服務將消息發送給iPhone應用程序。

不管是iPhone客戶端跟APNS,仍是Provider和APNS都須要經過證書進行鏈接的: ide

 

iOS推送通知 push 02 iOS開發:推送通知簡述及開發實踐 (重要!)

 

圖中, 測試

1. Provider是指某個iPhone軟件的Push服務器,是咱們將要開發的服務器。 加密

2. APNS 是Apple Push Notification Service(Apple Push服務器)的縮寫,是蘋果的服務器。 spa

上圖能夠分爲三個階段:

第一階段:推送服務器(provider)把要發送的消息、目的iPhone的標識打包,發給APNS;

第二階段:APNS在自身的已註冊Push服務的iPhone列表中,查找有相應標識的iPhone,並把消息發到iPhone;

第三階段:iPhone把發來的消息傳遞給相應的應用程序,而且按照設定彈出Push通知。

三.開發證書和推送證書的配置

 

1. 使用開發者賬號登陸IOS Provisioning ,選擇或新建一個App Id,這裏以「info.luoyl.iostest」爲例

iOS推送通知 push 1 iOS開發:推送通知簡述及開發實踐 (重要!)

 

2. 建立完後,進入App Id列表,能夠看到新建的App Id默認是沒有激活推送功能的,點擊Configure連接,進入推送功能激活頁面:

iOS推送通知 push 2 iOS開發:推送通知簡述及開發實踐 (重要!)

 

3. 在「Enable for Apple Push Notification service」選項上打勾,而後在行點「configure」按鈕:

iOS推送通知 push 3 iOS開發:推送通知簡述及開發實踐 (重要!)

 

4. 此時會彈出一窗口,點「continue」

iOS推送通知 push 4 iOS開發:推送通知簡述及開發實踐 (重要!)


5. 彈出證書上傳頁面,證書選擇事先作好的「CertificateSigningRequest.certSigningRequest」,而後點「Generate」按鈕;

iOS推送通知 push 5 iOS開發:推送通知簡述及開發實踐 (重要!)

 

6. 接下來會有「Your APNs SSL Certificate has been generated.」提示,點「continue」:

 

 

 

iOS推送通知 push 6 iOS開發:推送通知簡述及開發實踐 (重要!)

 

7. 下載剛生成的證書「aps_development.cer」到電腦:

iOS推送通知 push 7 iOS開發:推送通知簡述及開發實踐 (重要!)

 

8. 至此,appid的Development Push SSL Certificate已經變成「Enabled」狀態了:

iOS推送通知 push 8 iOS開發:推送通知簡述及開發實踐 (重要!)

9. 製做一開發者測試證書,appid指定爲「info.luoyl.iostest」, 下載後雙擊安裝到電腦上:

iOS推送通知 push 9 iOS開發:推送通知簡述及開發實踐 (重要!)

 

10. 雙擊在步驟7下載的「aps_development.cer」安裝到keychain Access上:

iOS推送通知 push 10 iOS開發:推送通知簡述及開發實踐 (重要!)

 

11. 選中push Services證書,右鍵導出證書爲我的信息交換(.p12)格式文件,這裏我命名爲「aps_development.p12」,點存儲時會彈出一個密碼設置窗口,可留空不填:

iOS推送通知 push 11 iOS開發:推送通知簡述及開發實踐 (重要!)

 

12. 在終端執行下面的命令,把剛纔導出的我的信息交換(.p12)格式文件加密轉換成推送服務器的推送證書:

 

  1. openssl pkcs12 -clcerts -nokeys -out cert.pem -in aps_development.p12
  2. openssl pkcs12 -nocerts -out key.pem -in aps_development.p12
  3. openssl rsa -in key.pem -out key.unencrypted.pem
  4. cat cert.pem key.unencrypted.pem > iostest_push_dev.pems

iOS推送通知 push 12 iOS開發:推送通知簡述及開發實踐 (重要!)

 

上面的命令在執行時有4處是須要輸入密碼的,其中1和2直接回車,3必須設定一個key如「push」,在4處輸入3設定的key 「push」;
命令執行完後生成的「iostest_push_dev.pem」就是咱們推送服務器要使用的推送證書;

通過以上步驟的配置,已經完成了開發推送功能所須要的條件了,接下來將會新建一個ios應用來體驗完成推送功能,在ios應用須要實現的接口。

 

.開發帶有推送功能的IOS應用

爲使應用能支持推送功能,咱們的項目配置時要注意:

  1. Bundle Identifier、Code Signing指定的開發證書綁定的AppId要和推送證書綁定的AppId一致(見下圖);
  2.  若是項目中的開發證書在AppId激活推送功能前已經建立了,這時必須從新生成一個。支持推送功能的開發證書會比舊證書多出一項名爲 「aps-environment」的受權串,若是繼續使用舊證書,在程序啓動嘗試註冊推送功能時會出現「 未找到應用程序的「aps-environment」的權利字符串 」的錯誤;
  3. 測試須要用真機,模擬器不支持。

iOS推送通知 push 13 iOS開發:推送通知簡述及開發實踐 (重要!)

 

iOS推送通知 push 14 iOS開發:推送通知簡述及開發實踐 (重要!)

 

在代碼方面,推送的註冊、監聽和處理都集中在AppDelegate類裏:

1.(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
在該方法體裏主要實現兩個功能:
一是完成推送功能的註冊請求,即在程序啓動時彈出是否使用推送功能;
二是實現的程序啓動是經過推送消息窗口觸發的,在這裏能夠處理推送內容;

  1. 1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
  2. {
  3.     self.window = [[[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]] autorelease];
  4.     // Override point for customization after application launch.
  5.     self.viewController = [[[ViewController alloc] init] autorelease];
  6.     self.window.rootViewController = self.viewController;
  7.     [self.window setBackgroundColor:[UIColor colorWithPatternImage:[UIImage imageNamed:@"background.png"]]];
  8.     [self.window makeKeyAndVisible];
  9.     
  10.     [[UIApplication sharedApplication] registerForRemoteNotificationTypes:(UIRemoteNotificationTypeAlert | UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound)];
  11.     //判斷程序是否是由推送服務完成的
  12.     if (launchOptions) {
  13.         NSDictionary* pushNotificationKey = [launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
  14.         if (pushNotificationKey) {
  15.             UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@」推送通知」
  16.                                                            message:@」這是經過推送窗口啓動的程序,你能夠在這裏處理推送內容」
  17.                                                           delegate:nil
  18.                                                  cancelButtonTitle:@」知道了」
  19.                                                  otherButtonTitles:nil, nil];
  20.             [alert show];
  21.             [alert release];
  22.         }
  23.     }
  24.     return YES;
  25. }

2. 接收從蘋果服務器返回的惟一的設備token,該token是推送服務器發送推送消息的依據,因此須要發送回推送服務器保存

 

  1. - (void)application:(UIApplication *)app didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
  2.     NSString* token = [NSString stringWithFormat:@"%@",deviceToken];
  3.     NSLog(@」apns -> 生成的devToken:%@」, token);
  4.     //把deviceToken發送到咱們的推送服務器
  5.     DeviceSender* sender = [[[DeviceSender alloc]initWithDelegate:self ]autorelease];
  6.     [sender sendDeviceToPushServer:token ];
  7. }
  8. 3.接收註冊推送通知功能時出現的錯誤,並作相關處理:

     

    1. - (void)application:(UIApplication *)app didFailToRegisterForRemoteNotificationsWithError:(NSError *)err {
    2.     NSLog(@」apns -> 註冊推送功能時發生錯誤, 錯誤信息:\n %@」, err);
    3. }

     

    4. 接收到推送消息,解析處理

     

    1. - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
    2. {
    3.     NSLog(@」\napns -> didReceiveRemoteNotification,Receive Data:\n%@」, userInfo);
    4.     //把icon上的標記數字設置爲0,
    5.     application.applicationIconBadgeNumber = 0;
    6.     if ([[userInfo objectForKey:@"aps"] objectForKey:@」alert」]!=NULL) {
    7.         UIAlertView* alert = [[UIAlertView alloc] initWithTitle:@」**推送消息**」
    8.                                                         message:[[userInfo objectForKey:@"aps"] objectForKey:@」alert」]
    9.                                                        delegate:self
    10.                                               cancelButtonTitle:@」關閉」
    11.                                               otherButtonTitles:@」處理推送內容」,nil];
    12.         alert.tag = alert_tag_push;
    13.         [alert show];
    14.     }
    15. }

    經過上面的代碼,基本推送功能的開發已經完成了。最後附件是一個簡單的推送demo和demo運行截圖,該demo須要推送服務器支持,請自行解決。

    iOS推送通知 push 15 iOS開發:推送通知簡述及開發實踐 (重要!)

相關文章
相關標籤/搜索