轉自: http://luoyl.info/blog/2012/02/apple_push_notification_guide/
一.關於推播通知
推播通知,也被叫做遠程通知,是在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.pems
上面的命令在執行時有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:@"推播通知" &nbs