原文地址:原文地址
一.關(guān)于推送通知
推送通知奶卓,也被叫做遠(yuǎn)程通知一疯,是在iOS3.0以后被引入的功能。是當(dāng)程序沒有啟動或不在前臺運行時夺姑,告訴用戶有新消息的一種途徑墩邀,是從外部服務(wù)器發(fā)送到應(yīng)用程序上的。一般說來盏浙,當(dāng)要顯示消息或下載數(shù)據(jù)的時候眉睹,通知是由遠(yuǎn)程服務(wù)器(程序的提供者)發(fā)送,然后通過蘋果的推送通知服務(wù)(Apple Push Notification Service废膘,簡稱apns)推送到設(shè)備的程序上竹海。
推送的新消息可能是一條信息、一項即將到期的日程或是一份遠(yuǎn)程服務(wù)器上的新數(shù)據(jù)丐黄。在系統(tǒng)上展現(xiàn)的時候斋配,可以顯示警告信息或在程序icon上顯示數(shù)字,同時灌闺,也可以播放警告音艰争。一旦用戶注意到程序有新的信息、時間或是數(shù)據(jù)桂对,他們可以運行程序并訪問新的內(nèi)容甩卓。也可以選擇忽略通知,這時程序?qū)⒉粫患せ睢?br>
iPhone, iPad和iPod touch上同一時刻只有一個app在前臺運行蕉斜。大多數(shù)程序在后臺運行的時候逾柿,可以對某些用戶感興趣的內(nèi)容做出回應(yīng)(定時、或數(shù)據(jù)等)宅此。推送通知能讓程序在這些事件發(fā)生的時候通知用戶机错。
作為提供者為程序開發(fā)和部署推送通知,必須通過iOS Developer Program Portal獲得SSL證書父腕。每個證書限用于一個程序毡熏,使用程序的bundle ID作為標(biāo)識。證書有兩種用途的:一種是針對sandbox(用于開發(fā)和測試)侣诵,另外一種針對發(fā)布產(chǎn)品痢法。這兩種運行環(huán)境擁有為各自指定的IP地址并且需要不同的證書狱窘。還必須為兩種不同的環(huán)境獲取各自的provisioning profiles。
APNS提供了兩項基本的服務(wù):消息推送和反饋服務(wù)财搁。
消息推送:使用流式TCP套接字將推送通知作為二進(jìn)制數(shù)據(jù)發(fā)送給APNs蘸炸。消息推送有分別針對開發(fā)和測試用的sandbox、發(fā)布產(chǎn)品的兩個接口尖奔,每個都有各自的地址和端口搭儒。不管用哪個接口,都需要通過TLS或SSL提茁,使用SSL證書來建立一個安全的信道淹禾。提供者編制通知信息,然后通過這個信道將其發(fā)送給APNs茴扁。注:sandbox: gateway.sandbox.push.apple.com:219產(chǎn)品接口:gateway.push.apple.com:2195
反饋服務(wù):可以得到針對某個程序的發(fā)送失敗記錄铃岔。提供者應(yīng)該使用反饋服務(wù)周期性檢查哪些設(shè)備一直收不到通知,不需要重復(fù)發(fā)送通知到這些設(shè)備峭火,降低推送服務(wù)器的負(fù)擔(dān)毁习。注:sandbox:feedback.push.apple.com:2196產(chǎn)品接口:feedback.sandbox.push.apple.com:2196
二.Apple Push Notification的工作機(jī)制
下面是一個完整推送流程圖從上圖,我們可以看到卖丸。
首先是應(yīng)用程序注冊消息推送纺且。
IOS跟APNS Server要deviceToken。應(yīng)用程序接受deviceToken稍浆。
應(yīng)用程序?qū)eviceToken發(fā)送給PUSH服務(wù)端程序(Provider)载碌。
服務(wù)端程序向APNS服務(wù)發(fā)送消息。
APNS服務(wù)將消息發(fā)送給iPhone應(yīng)用程序衅枫。
圖中嫁艇,
- Provider是指某個iPhone軟件的Push服務(wù)器,是我們將要開發(fā)的服務(wù)器为鳄。
- APNS 是Apple Push Notification Service(Apple Push服務(wù)器)的縮寫裳仆,是蘋果的服務(wù)器腕让。
上圖可以分為三個階段:
第一階段:推送服務(wù)器(provider)把要發(fā)送的消息孤钦、目的iPhone的標(biāo)識打包,發(fā)給APNS纯丸;
第二階段:APNS在自身的已注冊Push服務(wù)的iPhone列表中偏形,查找有相應(yīng)標(biāo)識的iPhone,并把消息發(fā)到iPhone觉鼻;
第三階段:iPhone把發(fā)來的消息傳遞給相應(yīng)的應(yīng)用程序俊扭,并且按照設(shè)定彈出Push通知。
三.開發(fā)證書和推送證書的配置
-
使用開發(fā)者帳號登錄IOS Provisioning ,選擇或新建一個App Id坠陈,這里以“info.luoyl.iostest”為例
-
創(chuàng)建完后萨惑,進(jìn)入App Id列表捐康,可以看到新建的App Id默認(rèn)是沒有激活推送功能的,點擊Configure鏈接庸蔼,進(jìn)入推送功能激活頁面:
-
在“Enable for Apple Push Notification service”選項上打勾解总,然后在行點“configure”按鈕:
-
此時會彈出一窗口,點“continue”
-
彈出證書上傳頁面姐仅,證書選擇事先做好的“CertificateSigningRequest.certSigningRequest”花枫,然后點“Generate”按鈕;
-
接下來會有“Your APNs SSL Certificate has been generated.”提示掏膏,點“continue”:
-
下載剛生成的證書“aps_development.cer”到電腦:
-
至此劳翰,appid的Development Push SSL Certificate已經(jīng)變成“Enabled”狀態(tài)了:
-
制作一開發(fā)者測試證書,appid指定為“info.luoyl.iostest”, 下載后雙擊安裝到電腦上:
-
雙擊在步驟7下載的“aps_development.cer”安裝到keychain Access上:
-
選中push Services證書馒疹,右鍵導(dǎo)出證書為個人信息交換(.p12)格式文件佳簸,這里我命名為“aps_development.p12”,點存儲時會彈出一個密碼設(shè)置窗口,可留空不填:
- 在終端執(zhí)行下面的命令行冰,把剛才導(dǎo)出的個人信息交換(.p12)格式文件加密轉(zhuǎn)換成推送服務(wù)器的推送證書:
[cpp] view plain copy
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
[圖片上傳中溺蕉。。悼做。(14)]
上面的命令在執(zhí)行時有4處是需要輸入密碼的疯特,其中1和2直接回車,3必須設(shè)定一個key如“push”,在4處輸入3設(shè)定的key “push”;命令執(zhí)行完后生成的“iostest_push_dev.pem”就是我們推送服務(wù)器要使用的推送證書肛走;
經(jīng)過以上步驟的配置漓雅,已經(jīng)完成了開發(fā)推送功能所需要的條件了,接下來將會新建一個ios應(yīng)用來體驗完成推送功能朽色,在ios應(yīng)用需要實現(xiàn)的接口邻吞。
四.開發(fā)帶有推送功能的IOS應(yīng)用
為使應(yīng)用能支持推送功能,我們的項目配置時要注意:
Bundle Identifier葫男、Code Signing指定的開發(fā)證書綁定的AppId要和推送證書綁定的AppId一致(見下圖)抱冷;
如果項目中的開發(fā)證書在AppId激活推送功能前已經(jīng)創(chuàng)建了,這時必須重新生成一個梢褐。支持推送功能的開發(fā)證書會比舊證書多出一項名為 “aps-environment”的授權(quán)串旺遮,如果繼續(xù)使用舊證書,在程序啟動嘗試注冊推送功能時會出現(xiàn)“ 未找到應(yīng)用程序的“aps-environment”的權(quán)利字符串 ”的錯誤盈咳;
測試需要用真機(jī)耿眉,模擬器不支持。
在代碼方面鱼响,推送的注冊鸣剪、監(jiān)聽和處理都集中在AppDelegate類里:
1.(BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
在該方法體里主要實現(xiàn)兩個功能:
- 一是完成推送功能的注冊請求,即在程序啟動時彈出是否使用推送功能;
- 二是實現(xiàn)的程序啟動是通過推送消息窗口觸發(fā)的筐骇,在這里可以處理推送內(nèi)容债鸡;
- (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)];
//判斷程序是不是由推送服務(wù)完成的
if (launchOptions) {
NSDictionary* pushNotificationKey = [launchOptions objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey];
if (pushNotificationKey) {
UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"推送通知"
message:@"這是通過推送窗口啟動的程序,你可以在這里處理推送內(nèi)容"
delegate:nil
cancelButtonTitle:@"知道了"
otherButtonTitles:nil, nil];
[alert show];
[alert release];
}
}
return YES;
}
2.接收從蘋果服務(wù)器返回的唯一的設(shè)備token铛纬,該token是推送服務(wù)器發(fā)送推送消息的依據(jù)娘锁,所以需要發(fā)送回推送服務(wù)器保存
- (void)application:(UIApplication *)app didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken {
NSString* token = [NSString stringWithFormat:@"%@",deviceToken];
NSLog(@"apns -> 生成的devToken:%@", token);
//把deviceToken發(fā)送到我們的推送服務(wù)器
DeviceSender* sender = [[[DeviceSender alloc]initWithDelegate:self ]autorelease];
[sender sendDeviceToPushServer:token ];
}
3.接收注冊推送通知功能時出現(xiàn)的錯誤,并做相關(guān)處理:
- (void)application:(UIApplication *)app didFailToRegisterForRemoteNotificationsWithError:(NSError *)err {
NSLog(@"apns -> 注冊推送功能時發(fā)生錯誤饺鹃, 錯誤信息:\n %@", err);
}
4.接收到推送消息莫秆,解析處理
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo
{
NSLog(@"\napns -> didReceiveRemoteNotification,Receive Data:\n%@", userInfo);
//把icon上的標(biāo)記數(shù)字設(shè)置為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:@"關(guān)閉"
otherButtonTitles:@"處理推送內(nèi)容",nil];
alert.tag = alert_tag_push;
[alert show];
}
}
通過上面的代碼,基本推送功能的開發(fā)已經(jīng)完成了悔详。最后附件上面代碼中所需用到的DeviceSender的類文件镊屎,需要將其頭文件導(dǎo)入到AppDelegate中。下面是DeviceSender類的.h和.m文件的下載地址:
http://download.csdn.net/detail/enuola/5100134