IOS推送消息是许多IOS应用都具有的功能,最近也在研究这个功能,参考了不少资料终于搞定了,下面就把步骤拿出来分享下:php
iOS消息推送的工做机制能够简单的用下图来归纳:数据库
Provider是指某个iPhone软件的Push服务器,APNS是Apple Push Notification Service的缩写,是苹果的服务器。json
上图能够分为三个阶段:xcode
第一阶段:应用程序把要发送的消息、目的iPhone的标识打包,发给APNS。 安全
第二阶段:APNS在自身的已注册Push服务的iPhone列表中,查找有相应标识的iPhone,并把消息发送到iPhone。 服务器
第三阶段:iPhone把发来的消息传递给相应的应用程序,而且按照设定弹出Push通知。app
从上图咱们能够看到:iphone
一、应用程序注册消息推送。socket
二、iOS从APNS Server获取device token,应用程序接收device token。ide
三、应用程序将device token发送给PUSH服务端程序。
四、服务端程序向APNS服务发送消息。
五、APNS服务将消息发送给iPhone应用程序。
不管是iPhone客户端和APNS,仍是Provider和APNS,都须要经过证书进行链接。
下面我介绍一下几种用到的证书。
1、CSR文件
一、生成Certificate Signing Request(CSR)
二、填写你的邮箱和经常使用名称,并选择保存到硬盘。
点击继续:
这样就在本地生成了一个Push.certSigningRequest文件。
2、p12文件
一、导出密钥。
二、输入你的密码。
这样就生成了一个Push.p12文件。
3、SSL certificate文件
一、用你付过费的账号登陆到iOS Provisioning Portal,并新建一个App ID,这个过程能够参考:iOS应用的真机调试,这样就会生成下面这条记录:
二、点击右侧的Configure:
三、点击Development Push SSL Certificate一行后的Configure:
四、点击Continue:
五、选择前面生成好的Push.certSigningRequest文件,点击Generate,出现以下所示的页面:
六、点击Continue:
七、点击Download,并将文件命名为aps_developer_identity.cer。
八、点击Done,你会发现状态变成了Enabled:
到如今为止,咱们已经生成了三个文件:
一、Push.certSigningRequest
二、Push.p12
三、aps_developer_identity.cer
双击aps_developer_dientity.cer 注册到你的钥匙串中,这样你的钥匙串中就会有
2、准备profile证书,由于推送消息只能再真机上测试,因此要建一个profile证书
点击"new profile"为上面新建的APP ID建个profile ,成功以后下载*_Dev_Profile.mobileprovision
双击将其加入到xcode 的Provisioning Profiles 中,这里有一点要注意,再将这个加入xcode以前若是以前已经加入过必定要把以前加入的删掉,若是有多个的话会出错。
3、工程代码
到这里证书已经准备完毕,接下来,咱们在xcode中新建一个测试工程,注意设置工程的Bundle Identifier必须与上面建的APP ID 里的相同
在didFinishLaunchingWithOptions 中加入一下代码
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
[self.window makeKeyAndVisible];
[[UIApplication sharedApplication] registerForRemoteNotificationTypes: UIRemoteNotificationTypeBadge | UIRemoteNotificationTypeSound | UIRemoteNotificationTypeAlert];
return YES;
}
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)pToken {
NSLog(@"regisger success:%@", pToken);
//注册成功,将deviceToken保存到应用服务器数据库中
}
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo{
// 处理推送消息
UIAlertView *alert=[[UIAlertView alloc]initWithTitle:@"通知" message:@"个人信息" delegate:self cancelButtonTitle:@"取消" otherButtonTitles:nil, nil];
[alert show];
[alert release];
NSLog(@"%@", userInfo);
}
- (void)application:(UIApplication *)application didFailToRegisterForRemoteNotificationsWithError:(NSError *)error {
NSLog(@"Regist fail%@",error);
}
到这里一切顺利的话咱们就能够在真机运行了,注册成功咱们会获得iphone 的deviceToken,
My token is:
<740f4707 bebcf74f 9b7c25d4 8e335894 5f6aa01d a5ddb387 462c7eaf 61bb78ad>
4、在应用服务器采用php的方式将消息推送给APNS,
一、php链接APNS也是须要证书的,还记得咱们上面得到的几个证书吗?打开终端,对上面的证书作以下处理,
cd 进入证书所在目录
把.cer文件转换成.pem文件:
$ openssl x509 -in aps_developer_identity.cer -inform der
-out PushChatCert.pem
把私钥Push.p12文件转换成.pem文件:
$ openssl pkcs12 -nocerts -out PushChatKey.pem -in Push.p12
Enter Import Password:
MAC verified OK
Enter PEM pass phrase:
Verifying – Enter PEM pass phrase:
你首先须要为.p12文件输入passphrase密码短语,这样OpenSSL能够读它。而后你须要键入一个新的密码短语来加密PEM文件。仍是使用”pushchat”来做为PEM的密码短语。你须要选择一些更安全的密码短语。
注意:若是你没有键入一个PEM passphrase,OpenSSL将不会返回一个错误信息,可是产生的.pem文件里面将不会含有私钥。
最后。把私钥和证书整合到一个.pem文件里:
$ cat PushChatCert.pem PushChatKey.pem > ck.pem
为了测试证书是否工做,执行下面的命令:
$ telnet gateway.sandbox.push.apple.com 2195
Trying 17.172.232.226…
Connected to gateway.sandbox.push-apple.com.akadns.net.
Escape character is ‘^]’.
它将尝试发送一个规则的,不加密的链接到APNS服务。若是你看到上面的反馈,那说明你的MAC可以到达APNS。按下Ctrl+C 关闭链接。若是获得一个错误信息,那么你须要确保你的防火墙容许2195端口。
而后再次链接,此次用咱们的SSL证书和私钥来设置一个安全的链接:
$ openssl s_client -connect gateway.sandbox.push.apple.com:2195
-cert PushChatCert.pem -key PushChatKey.pem
Enter pass phrase for PushChatKey.pem:
你会看到一个完整的输出,让你明白OpenSSL在后台作什么。若是链接是成功的,你能够键入一些字符。当你按下回车后,服务就会断开链接。若是在创建链接时有问题,OpenSSL将会给你一个错误消息,
ck.pem文件就是咱们须要获得php链接APNS 的文件,将ck.pem和push.php放入同一目录上传到服务器,push.php的代码以下:
<?php
// 这里是咱们上面获得的deviceToken,直接复制过来(记得去掉空格)
$deviceToken = '740f4707bebcf74f 9b7c25d4 8e3358945f6aa01da5ddb387462c7eaf 61bb78ad';
// Put your private key's passphrase here:
$passphrase = 'abc123456';
// 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);
?>
接下来咱们访问http://localhost/push/push.php
iphone就会接收到一条推送消息了,若是有问题的话就检查上面的操做步骤,特别是加红的部分
另外去除标记的方法为,在viewDidApper中加入
int badge = [UIApplication sharedApplication].applicationIconBadgeNumber;
if(badge > 0)
{
badge--;
[UIApplication sharedApplication].applicationIconBadgeNumber = badge;
}
给一个参考地址:http://article.ityran.com/archives/194