iOS消息推送的工作机制可以简单的用下图来概括:
Provider是指某个iPhone软件的Push服务器,APNS是Apple Push Notification Service的缩写,是苹果的服务器.
上图可以分为三个阶段:
第一阶段:应用程序把要发送的消息、目的iPhone的标识打包,发给APNS.
第二阶段:APNS在自身的已注册Push服务的iPhone列表中,查找有相应标识的iPhone,并把消息发送到iPhone.
第三阶段:iPhone把发来的消息传递给相应的应用程序,并且按照设定弹出Push通知.
①.、应用程序注册消息推送.
下面我介绍一下几种用到的证书.
第一段:CSR文件
①.、生成Certificate Signing Request(CSR)
这样就在本地生成了一个Push.certSigningRequest文件.
①.、导出密钥.
第三段:SSL certificate文件
①.、用你付过费的帐号登录到iOS Provisioning Portal,并新建一个App ID,这个过程可以参考:iOS应用的真机调试,这样就会生成下面这条记录:
到现在为止,我们已经生成了三个文件:
①.、Push.certSigningRequest
双击aps_developer_dientity.cer 注册到你的钥匙串中,这样你的钥匙串中就会有
第二段:准备profile证书,因为推送消息只能再真机上测试,所以要建一个profile证书
双击将其加入到xcode 的Provisioning Profiles 中,这里有一点要注意,再将这个加入xcode之前如果之前已经加入过一定要把之前加入的删掉,如果有多个的话会出错.
第三段:工程代码
到这里证书已经准备完毕,此时此刻呢,我们在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:selfcancelButtonTitle:@"取消" 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:
第四段:在应用服务器采用php的方式将消息推送给APNS,
①.、php连接APNS也是需要证书的,还记得我们上面获得的几个证书吗?打开终端,对上面的证书做如下处理,
cd 进入证书所在目录
把.cer文件转换成.pem文件:
-out PushChatCert.pem
Enter Import Password:
MAC verified OK
Enter PEM pass phrase:
Verifying – Enter PEM pass phrase:
注意:如果你没有键入一个PEM passphrase,OpenSSL将不会返回一个错误信息,但是产生的.pem文件里面将不会含有私钥.
最后.把私钥和证书整合到一个.pem文件里:
$ cat PushChatCert.pem PushChatKey.pem ck.pem
为了测试证书是否工作,执行下面的命令:
Connected to gateway.sandbox.push-apple.com.akadns.net.
Escape character is '^]'.
然后再次连接,这次用我们的SSL证书和私钥来设置一个安全的连接:
-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,直接复制过来(记得去掉空格)
// Put your private key's passphrase here:
// 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
//这个为正是的发布地址
//这个是沙盒测试地址,发布到appstore后记得修改哦
$fp = stream_socket_client(
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
// 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);
此时此刻呢我们访问
iphone就会接收到一条推送消息了,如果有问题的话就检查上面的操作步骤,特别是加红的部分
另外去除标记的方法为,在viewDidApper中加入
int badge = [UIApplication sharedApplication].applicationIconBadgeNumber;
if(badge 0)
badge--;
[UIApplication sharedApplication].applicationIconBadgeNumber = badge;
①.,签到之后,班级,姓名,学号这些都是可以立即获取到的.
可以使用缓存技术,先写到内存中,在同步到数据库中,memcached,你试试
那你可以在插入数据库那里设置监听呀,一有插入就往Android推送消息.
不然你只能通过每隔一段时间刷新一下读取数据库,看看有没有更新,有就推送,没有就继续监听,这个你可以自己设置更新时间.
第一段:socket协议的简介
WebSocket是什么,有什么优点
WebSocket是一个持久化的协议,这是相对于http非持久化来说的.应用层协议
举个简单的例子,http1.0的生命周期是以request作为界定的,也就是一个request,一个response,对于http来说,本次client与server的会话到此结束;而在http1.1中,稍微有所改进,即添加了keep-alive,也就是在一个http连接中可以进行多个request请求和多个response接受操作.然而在实时通信中,并没有多大的作用,http只能由client发起请求,server才能返回信息,即server不能主动向client推送信息,无法满足实时通信的要求.而WebSocket可以进行持久化连接,即client只需进行一次握手,成功后即可持续进行数据通信,值得关注的是WebSocket实现client与server之间全双工通信,即server端有数据更新时可以主动推送给client端.
第二段:介绍client与server之间的socket连接原理
①.、下面是一个演示client和server之间建立WebSocket连接时握手部分
a、client建立WebSocket时向服务器端请求的信息
GET /chat HTTP/1.1
Host: server.example.com
Upgrade: websocket //告诉服务器现在发送的是WebSocket协议
Connection: Upgrade
Sec-WebSocket-Protocol: chat, superchat
Origin:
b、服务器获取到client请求的信息后,根据WebSocket协议对数据进行处理并返回,其中要对Sec-WebSocket-Key进行加密等操作
HTTP/1.1 101 Switching Protocols
Sec-WebSocket-Protocol: chat
第三段:PHP中建立websocket的过程讲解
SocketService.php:
web.html:
后端代码
push.php
use?Workerman\Worker;
require_once?'./Workerman/Autoloader.php';
//?这里进程数必须设置为1
$worker-count?=?1;
//?worker进程启动后建立一个内部通讯端口
$worker-onWorkerStart?=?function($worker)
//?开启一个内部端口,方便内部系统推送数据,Text协议格式?文本◆换行符
$inner_text_worker-onMessage?=?function($connection,?$buffer)
global?$worker;
//?$data数组格式,里面有uid,表示向那个uid的页面推送数据
$data?=?json_decode($buffer,?true);
$uid?=?$data['uid'];
//?通过workerman,向uid的页面推送数据
$ret?=?sendMessageByUid($uid,?$buffer);
//?返回推送结果
$connection-send($ret?'ok'?:?'fail');
};
$inner_text_worker-listen();
//?新增加一个属性,用来保存uid到connection的映射
$worker-uidConnections?=?array();
$worker-onMessage?=?function($connection,?$data)use($worker)
if(!isset($connection-uid))
//?没验证的话把第一个包当做uid(这里为了方便演示,没做真正的验证)
$connection-uid?=?$data;
/*?保存uid到connection的映射,这样可以方便的通过uid查找connection,
*?实现针对特定uid推送数据
*/
$worker-uidConnections[$connection-uid]?=?$connection;
return;
$worker-onClose?=?function($connection)use($worker)
if(isset($connection-uid))
//?连接断开时删除映射
unset($worker-uidConnections[$connection-uid]);
//?向所有验证的用户推送数据
function?broadcast($message)
foreach($worker-uidConnections?as?$connection)
$connection-send($message);
//?针对uid推送数据
function?sendMessageByUid($uid,?$message)
if(isset($worker-uidConnections[$uid]))
$connection?=?$worker-uidConnections[$uid];
return?true;
return?false;
//?运行所有的worker(其实当前只定义了一个)
Worker::runAll();
启动后端服务
php?push.php?start?-d
前端接收推送的js代码
ws.onopen?=?function(){
var?uid?=?'uid1';
ws.send(uid);
ws.onmessage?=?function(e){
alert(e.data);
后端推送消息的代码
//?建立socket连接到内部推送端口
//?推送的数据,包含uid字段,表示是给这个uid推送
fwrite($client,?json_encode($data)."\n");
//?读取推送结果
这里的uid不一定是用户的id,也可以理解为任务id即 taskid
以上代码亲测可以直接使用