好久沒有寫過博客啦,今天就由本菜鳥給大家做一個(gè)簡單的IOSApp消息推送教程吧!一切從0開始芜繁,包括XCode6, IOS8, 以及蘋果開發(fā)者中心最新如何注冊應(yīng)用鳍咱,申請證書以及下載配置概要文件,相信很多剛開始接觸iOS的人會(huì)很想了解一下措嵌。(ps:網(wǎng)上看了一下雖然有很多講述推送的好教程躲叼,我也是看著一步步學(xué)會(huì)的,但是這些教程的時(shí)間都是去年或者更早時(shí)期的企巢,對引導(dǎo)新手來說不是很合適)
首先第一步當(dāng)然是介紹一下蘋果的推送機(jī)制(APNS)咯(ps:其實(shí)每一篇教程都有)枫慷,先來看一張?zhí)O果官方對其推送做出解釋的概要圖。
Provider是給你手機(jī)應(yīng)用發(fā)出推送消息的服務(wù)器浪规,而APNS(Apple Push Notification Service)則是蘋果消息推送服務(wù)器或听。你本地的服務(wù)器當(dāng)需要給應(yīng)用推送一條消息的時(shí)候,先要將消息發(fā)出到蘋果推送服務(wù)器笋婿,然后再由蘋果推送服務(wù)器將消息發(fā)到安裝了該應(yīng)用的手機(jī)誉裆。
接下來再看一張解釋圖:
根據(jù)上圖的邏輯我來給大家解釋一下:
1.你的iOS應(yīng)用需要去注冊APNS消息推送功能。
2.當(dāng)蘋果APNS推送服收到來自你應(yīng)用的注冊消息就會(huì)返回一串device token給你(很重要)
3.將應(yīng)用收到的device Token傳給你本地的Push服務(wù)器缸濒。
4.當(dāng)你需要為應(yīng)用推送消息的時(shí)候足丢,你本地的推送服務(wù)器會(huì)將消息,以及Device Token打包發(fā)送到蘋果的APNS服
5.APNS再將消息推送給目的iphone
1.從證書頒發(fā)機(jī)構(gòu)頒發(fā)證書
打開你mac的鑰匙串訪問:
然后點(diǎn)擊鑰匙串訪問
隨后它會(huì)彈出一個(gè)窗口用戶電子郵件信息就填寫你蘋果開發(fā)者賬號的名稱即可(應(yīng)該是一個(gè)郵件名稱)庇配,點(diǎn)擊保存到磁盤的選
項(xiàng)斩跌,點(diǎn)擊繼續(xù),顯示如下:
點(diǎn)擊存儲(chǔ)捞慌,文件名為:CertificateSigningRequest.certSigningRequest 隨后將他放在一個(gè)文件夾中我們?nèi)∶鹥ush吧耀鸦!
訪問蘋果開發(fā)者網(wǎng)址:https://developer.apple.com/
選中MemberCenter選項(xiàng),進(jìn)入登陸頁面啸澡,用你的蘋果開發(fā)者賬號登陸,過一會(huì)網(wǎng)頁就會(huì)自動(dòng)跳轉(zhuǎn)到下圖揭糕。
點(diǎn)擊紅色所選部分內(nèi)容進(jìn)行下一步的操作萝快。
選擇Certificates選項(xiàng),設(shè)置證書著角,如圖所示先解釋一下:
Development選項(xiàng)的作用顧名思義就是用來作為開發(fā)使用的證書揪漩,Production選項(xiàng)則是用來發(fā)布產(chǎn)品使用的,名稱很陌生是不是吏口,之
前的開發(fā)者網(wǎng)頁是沒有這一選項(xiàng)的奄容,可能是蘋果把他修改了,用這個(gè)名稱更加能讓人理解吧(字面上解釋就是產(chǎn)品么)产徊。兩個(gè)選項(xiàng)生
成證書的步驟是一樣的昂勒,現(xiàn)在我們使用開發(fā)者的選項(xiàng)進(jìn)行證書的制作,步驟如下:選擇Development選項(xiàng):
點(diǎn)擊上面的加號選項(xiàng)舟铜,
選擇APNS選項(xiàng)(開發(fā)么當(dāng)然是在沙盒環(huán)境下了戈盈,模擬真實(shí)情況),然后Continue
這個(gè)AppID我們在下一部分講如何生成谆刨,現(xiàn)在我用的是已經(jīng)生成好的一個(gè)應(yīng)用ID塘娶,繼續(xù)Continue
這邊就要選擇在鑰匙串訪問環(huán)節(jié)下載下來的CertificateSigningRequest.certSigningRequest文件了,選擇并生成:
點(diǎn)擊下載痊夭,得到aps_development .cer刁岸,保存到push文件中去。
新建一個(gè)AppID,選擇網(wǎng)頁上的AppIDs,然后點(diǎn)擊右上角的 “加號”
App的取名只要按照蘋果要求的就可以了
然后BundleID是比較重要的她我,在提交審核以及測試(蘋果的TestFlight)和付費(fèi)環(huán)節(jié)都需要用到虹曙,也只需按照蘋果要求來寫就好了。
接下來就是對你的應(yīng)用需要使用蘋果的哪些服務(wù)進(jìn)行選擇就行了番舆,例如廣告酝碳,游戲中心,推送恨狈,付費(fèi)等等情況疏哗。
最后選擇“Submit”選項(xiàng),在下一個(gè)界面中選擇“done”選項(xiàng)拴事,這樣我們設(shè)置AppID的步驟我們就完成了沃斤。
第五部分:生成Provisioning Profiles
這個(gè)配置概要文件分為兩種圣蝎,一種是為開發(fā)使用的刃宵,還有一種則是為發(fā)布到appStore上面。
創(chuàng)建發(fā)布版的ProvisioningProfile與開發(fā)版的流程相同徘公,點(diǎn)擊Development然后點(diǎn)擊右上角的加號
會(huì)進(jìn)入選擇何種配置概要文件的界面:
我們現(xiàn)在時(shí)測試牲证,所以選擇“IOS App Development”的選項(xiàng),在下面的Distribution發(fā)布選項(xiàng)中有兩個(gè)選擇关面,“App Store”以及
“Ad hoc”,你可以根據(jù)下面的描述選擇你發(fā)布所需的選項(xiàng)坦袍。點(diǎn)擊Continue進(jìn)入下一步十厢。
選擇你上一步創(chuàng)建的AppID,點(diǎn)擊Continue 進(jìn)行下一步
選擇你的開發(fā)者賬號,Continue進(jìn)行下一步
在這一步上選擇你的設(shè)備(你只有在這一步上勾選了你的設(shè)備捂齐,你才能在設(shè)備上用這個(gè)簽名進(jìn)行調(diào)試)蛮放。關(guān)于如何將你的設(shè)備號添加
進(jìn)去也是非常簡單的,選擇左側(cè)的"Devices",然后點(diǎn)擊右上角的加號奠宜,在隨后出來的頁面上添加你設(shè)備的UUID(在XCode中可以查
看到)以及name( 可以隨便取包颁,自己看的懂就行)然后Register一下,照著流程走到最后一步就完成了压真。
好咱們繼續(xù)回到上面的Provisioning Profile配置環(huán)節(jié)娩嚼,當(dāng)你選好了你的設(shè)備后點(diǎn)擊“Continue”進(jìn)入下一頁,
輸入一個(gè)文件名(最好是起的能看懂是干嘛的滴肿,當(dāng)然也可以隨便起)岳悟,點(diǎn)擊“Generate”進(jìn)入下一個(gè)頁面,在這個(gè)頁面中就會(huì)有一個(gè)下
載按鈕讓你下載這個(gè)文件泼差,我們把它下載下來放在Push文件夾中贵少。
好了,前期的準(zhǔn)備工作都已經(jīng)做完了拴驮,現(xiàn)在讓我們開始推送吧4核病(吼吼)
首先雙擊我們生成的 “aps_development .cer” 文件,進(jìn)入鑰匙串訪問套啤,找到我們的專用秘鑰(根據(jù)在第二部分中從證書機(jī)構(gòu)頒發(fā)證書操作中填寫的常用名)
我在第二部分填寫的是“silicon”宽气,由于換了一臺(tái)mac之前安裝的沒有了,之前沒有截圖潜沦,所以隨便找了個(gè)圖給大家看一下萄涯,憑大家的聰明才智應(yīng)該不難理解吧。
然后右擊導(dǎo)出唆鸡,會(huì)彈出如下所示的圖涝影。
將他存儲(chǔ)到push文件夾中,命名為“push.p12”争占,在這一步中導(dǎo)出會(huì)讓你輸入密碼并驗(yàn)證燃逻,你可以自定義一個(gè)密碼,例如abc123
現(xiàn)在push文件夾中應(yīng)該有幾個(gè)文件“aps_development .cer” 臂痕,"push.p12"伯襟,“CertificateSigningRequest.certSigningRequest”以及
剛才下下來的配置概要文件。
接下來我們打開終端將他們生成.pem文件
1.把a(bǔ)ps_development .cer文件生成.pcm文件握童,cd到push文件夾下
2.把push.p12文件生成為.pem文件
上邊輸入的密碼則是你導(dǎo)出證書所設(shè)的密碼姆怪,即abc123.接著還會(huì)讓你輸入.pem文件的密碼,還是使用abc123好了,防止混淆稽揭。
這樣我們在push文件夾中就又得到了兩個(gè)文件俺附,PushChatCert.pem和PushChatKey.pem。
3.把PushChatCert.pem和PushChatKey.pem合并為一個(gè)pem文件溪掀,
在push文件夾中又多了一個(gè)ck.pem文件事镣,以上我們把需要使用的文件都準(zhǔn)備好了
接下來就要測試一下啦,是不是很激動(dòng)~
為了測試證書工作的狀況揪胃,我們可以使用“telnet gateway.sandbox.push.apple.com 2195”來檢測一下蛮浑,如果顯示下圖則表示成功了。
然后只嚣,我們使用我們生成的證書和私鑰來設(shè)置一個(gè)安全的鏈接去鏈接蘋果服務(wù)器
在終端輸入如下命令:openssl s_client -connect gateway.sandbox.push.apple.com:2195 -cert PushChatCert.pem -key PushChatKey.pem
需要輸入密碼(abc123 我們剛才所設(shè)置的)沮稚。
然后他會(huì)返回一系列的數(shù)據(jù),這里我就粘貼一部分啦:
CONNECTED(00000003)
depth=1 /C=US/O=Entrust, Inc./OU=www.entrust.NET/rpa is incorporated by reference/OU=(c) 2009 Entrust, Inc./CN=Entrust Certification Authority - L1C
verify error:num=20:unable to get local issuer certificate
verify return:0
---
Certificate chain
册舞。蕴掏。。调鲸。盛杰。(省略)
。藐石。即供。。于微。(省略)
逗嫡。。株依。驱证。。(省略)
Start Time: 1416389389
Timeout ? : 300 (sec)
Verify return code: 0 (ok)
---
測試就到這里啦恋腕。抹锄。。
1.建立推送項(xiàng)目
[objc]view plaincopy
//
//??AppDelegate.m
//??TestPushNotifiy
//
//??Created?by?silicon?on?14-10-30.
//??Copyright?(c)?2014年?silicon.?All?rights?reserved.
//
#import?"AppDelegate.h"
@implementationAppDelegate
@synthesizemainView?=?_mainView;
-?(BOOL)application:(UIApplication*)applicationdidFinishLaunchingWithOptions:(NSDictionary*)launchOptions
{
if([applicationrespondsToSelector:@selector(isRegisteredForRemoteNotifications)])
{
//IOS8
//創(chuàng)建UIUserNotificationSettings荠藤,并設(shè)置消息的顯示類類型
UIUserNotificationSettings*notiSettings?=?[UIUserNotificationSettingssettingsForTypes:(UIUserNotificationTypeBadge?|?UIUserNotificationTypeAlert?|?UIRemoteNotificationTypeSound)categories:nil];
[applicationregisterUserNotificationSettings:notiSettings];
}else{//?ios7
[applicationregisterForRemoteNotificationTypes:(UIRemoteNotificationTypeBadge???????????????????????????????????????|UIRemoteNotificationTypeSound??????????????????????????????????????|UIRemoteNotificationTypeAlert)];
}
self.window=?[[UIWindowalloc]initWithFrame:[[UIScreenmainScreen]bounds]];
//?Override?point?for?customization?after?application?launch.
self.window.backgroundColor=?[UIColorwhiteColor];
[self.windowmakeKeyAndVisible];
self.mainView=?[[MainViewControlleralloc]initWithNibName:@"MainViewController"bundle:nil];
self.window.rootViewController=self.mainView;
returnYES;
}
-?(void)application:(UIApplication*)applicationdidRegisterForRemoteNotificationsWithDeviceToken:(NSData*)pToken{
NSLog(@"---Token--%@",?pToken);
}
-?(void)application:(UIApplication*)applicationdidReceiveRemoteNotification:(NSDictionary*)userInfo{
NSLog(@"userInfo?==?%@",userInfo);
NSString*message?=?[[userInfoobjectForKey:@"aps"]objectForKey:@"alert"];
UIAlertView*alert?=?[[UIAlertViewalloc]initWithTitle:@"提示"message:messagedelegate:selfcancelButtonTitle:@"取消"otherButtonTitles:@"確定",nilnil];
[alertshow];
}
-?(void)application:(UIApplication*)applicationdidFailToRegisterForRemoteNotificationsWithError:(NSError*)error{
NSLog(@"Regist?fail%@",error);
}
-?(void)applicationWillResignActive:(UIApplication*)application
{
//?Sent?when?the?application?is?about?to?move?from?active?to?inactive?state.?This?can?occur?for?certain?types?of?temporary?interruptions?(such?as?an?incoming?phone?call?or?SMS?message)?or?when?the?user?quits?the?application?and?it?begins?the?transition?to?the?background?state.
//?Use?this?method?to?pause?ongoing?tasks,?disable?timers,?and?throttle?down?OpenGL?ES?frame?rates.?Games?should?use?this?method?to?pause?the?game.
}
-?(void)applicationDidEnterBackground:(UIApplication*)application
{
//?Use?this?method?to?release?shared?resources,?save?user?data,?invalidate?timers,?and?store?enough?application?state?information?to?restore?your?application?to?its?current?state?in?case?it?is?terminated?later.
//?If?your?application?supports?background?execution,?this?method?is?called?instead?of?applicationWillTerminate:?when?the?user?quits.
}
-?(void)applicationWillEnterForeground:(UIApplication*)application
{
//?Called?as?part?of?the?transition?from?the?background?to?the?inactive?state;?here?you?can?undo?many?of?the?changes?made?on?entering?the?background.
}
-?(void)applicationDidBecomeActive:(UIApplication*)application
{
//?Restart?any?tasks?that?were?paused?(or?not?yet?started)?while?the?application?was?inactive.?If?the?application?was?previously?in?the?background,?optionally?refresh?the?user?interface.
}
-?(void)applicationWillTerminate:(UIApplication*)application
{
//?Called?when?the?application?is?about?to?terminate.?Save?data?if?appropriate.?See?also?applicationDidEnterBackground:.
}
@end
在appdelegate.m中加入以上代碼伙单,
[objc]view plaincopy
if([applicationrespondsToSelector:@selector(isRegisteredForRemoteNotifications)])
{
//IOS8
//創(chuàng)建UIUserNotificationSettings,并設(shè)置消息的顯示類類型
UIUserNotificationSettings*notiSettings?=?[UIUserNotificationSettingssettingsForTypes:(UIUserNotificationTypeBadge?|?UIUserNotificationTypeAlert?|?UIRemoteNotificationTypeSound)categories:nil];
[applicationregisterUserNotificationSettings:notiSettings];
}else{//?ios7
[applicationregisterForRemoteNotificationTypes:(UIRemoteNotificationTypeBadge???????????????????????????????????????|UIRemoteNotificationTypeSound??????????????????????????????????????|UIRemoteNotificationTypeAlert)];
}
由于ios8的推送跟ios7及以下的不一樣哈肖,所以需要加判斷來注冊消息推送吻育。
函數(shù):
[objc]view plaincopy
-?(void)application:(UIApplication*)applicationdidRegisterForRemoteNotificationsWithDeviceToken:(NSData*)pToken{
NSLog(@"---Token--%@",?pToken);
}
會(huì)接收來自蘋果服務(wù)器給你返回的deviceToken,然后你需要將它添加到你本地的推送服務(wù)器上牡彻。(很重要扫沼,決定你的設(shè)備能不能接收到推送消息)。
[objc]view plaincopy
-?(void)application:(UIApplication*)applicationdidReceiveRemoteNotification:(NSDictionary*)userInfo{
NSLog(@"userInfo?==?%@",userInfo);
NSString*message?=?[[userInfoobjectForKey:@"aps"]objectForKey:@"alert"];
UIAlertView*alert?=?[[UIAlertViewalloc]initWithTitle:@"提示"message:messagedelegate:selfcancelButtonTitle:@"取消"otherButtonTitles:@"確定",nilnil];
[alertshow];
}
這個(gè)函數(shù)則是當(dāng)設(shè)備接收到來自蘋果推送服務(wù)器的消息時(shí)觸發(fā)的庄吼,用來顯示推送消息缎除。
[objc]view plaincopy
-?(void)application:(UIApplication*)applicationdidFailToRegisterForRemoteNotificationsWithError:(NSError*)error{
NSLog(@"Regist?fail%@",error);
}
當(dāng)注冊失敗時(shí),觸發(fā)此函數(shù)总寻。
2.PHP服務(wù)端
將simplepush.php這個(gè)推送腳本也放在push文件夾中
[php]view plaincopy
//???????????deviceToken???????????????
$deviceToken='c95f661371b085e2517b4c12cc76293522775e5fd9bb1dea17dd80fe85583b41';
//?Put?your?private?key's?passphrase?here:
$passphrase='abc123';
//?Put?your?alert?message?here:
$message='My?first?push?test!';
////////////////////////////////////////////////////////////////////////////////
$ctx=?stream_context_create();
stream_context_set_option($ctx,'ssl','local_cert','ck.pem');
stream_context_set_option($ctx,'ssl','passphrase',$passphrase);
//?Open?a?connection?to?the?APNS?server
//??????????
//$fp?=?stream_socket_client(?ssl://gateway.push.apple.com:2195?,?$err,?$errstr,?60,?//STREAM_CLIENT_CONNECT,?$ctx);
//?????????????appstore??????
$fp=?stream_socket_client(
'ssl://gateway.sandbox.push.apple.com:2195',$err,
$errstr,?60,?STREAM_CLIENT_CONNECT|STREAM_CLIENT_PERSISTENT,$ctx);
if(!$fp)
exit("Failed?to?connect:?$err?$errstr".?PHP_EOL);
echo'Connected?to?APNS'.?PHP_EOL;
//?Create?the?payload?body
$body['aps']?=array(
'alert'=>$message,
'sound'=>'default'
);
//?Encode?the?payload?as?JSON
$payload=?json_encode($body);
//?Build?the?binary?notification
$msg=chr(0)?.?pack('n',?32)?.?pack('H*',$deviceToken)?.?pack('n',strlen($payload))?.$payload;
//?Send?it?to?the?server
$result=?fwrite($fp,$msg,strlen($msg));
if(!$result)
echo'Message?not?delivered'.?PHP_EOL;
else
echo'Message?successfully?delivered'.?PHP_EOL;
//?Close?the?connection?to?the?server
fclose($fp);
?>
deviceToken填寫你接收到的token,passPhrase則填寫你的ck.pem設(shè)置的密碼器罐。
此刻就是見證奇跡的時(shí)候了,使用終端進(jìn)入到push文件夾,在終端輸入 php simplepush.php
推送就到這邊吧渐行!
感謝這篇博客的指導(dǎo):http://blog.csdn.net/showhilllee/article/details/8631734