您的位置:首页 > 编程语言 > PHP开发

php socket 模型及效率问题

2016-03-03 19:03 591 查看
// 创建套接字
socket_create();
// 绑定
socket_bind();
// 监听
socket_listen();

// 主体, 死循环
while(true){
// select模型, 取出可读套接字列表
socket_select();

foreach(sockets) {
// 如果是监听连接请求端口的套接字
if(is listensocket) {
// 接受请求
socket_accept();
}
else {
// 读取封包
socket­_recv();
// 处理用户动作 {问题就在这}
process();
}
}
}


php socket 模型及效率问题
呼呼安
3 票
呼呼安 70
// 创建套接字 socket_create(); // 绑定 socket_bind(); // 监听 socket_listen(); // 主体, 死循环 while(true){ // select模型, 取出可读套接字列表 socket_select(); foreach(sockets) { // 如果是监听连接请求端口的套接字 if(is listensocket) { // 接受请求 socket_accept(); } else { // 读取封包 socket­_recv(); // 处理用户动作 {问题就在这} process(); } } }

关于php做服务端的效率, 我想这里是个问题吧。
在每处理一个用户的动作的时候, 整个循环被阻塞在这里, 导致其他的连接不能接受, 其他用户的动作请求不能被处理。 直到process()函数返回才能处理下一个动作。
处理一个用户动作的时候, 可能还要涉及到数据库访问等, 可能比较耗时。
不知道有没有哪位大仙有解决办法呢。

我想的是, 如果能做一个动作队列, 这里的循环只负责接收数据, 然后把数据包保存在一个队列里, 就去接收下一个用户的封包。
另一个线程(不知道PHP怎么实现线程, 或是能不能实现)从队列中抽取每一个用户的动作请求来处理它。。

或着process()函数能不能做成非阻塞的, 让它可以立即返回。。

小弟初学,如有可笑之处,请见谅!

评论 (0) • 分享 • 链接 • 2012-07-24 
4个答案 票 数
冯义军
1 票
冯义军14.02K
最佳答案
这个问题我也遇到过到,比如打开两个终端,telnet socket服务端,如果其中一个不返回,另一个终端总是等待。最后使用fork子进程方式解决,大概代码如下,你可以参考下:

死循环部分

do {
$msgsock = socket_accept($socket);
$pid = pcntl_fork();
if($pid == -1) {
// fork error ;
}else if($pid) {
socket_close($msgsock);
}else {
$buf = socket_read($msgsock,1024,PHP_NORMAL_READ);
$ret = func($buf); //调用函数处理接收到的内容
socket_write($msgsock,$ret,strlen($ret));
socket_close($msgsock);
//posix_kill(posix_getpid(),0);
exit();
}
}while(true)

评论 (10) • 链接 • 2012-07-25
0
@冯义军 你好, 我在本地测试的时候, 可以用cmd运行php, 或是在网页上访问一下, 然后关闭网页就行了, php会一直执行, 然后端口就可以一直访问。。
但是我放服务器上(linux), 怎么让它一直执行啊, 我用网页访问它的时候可以连接, 如果关了网页, 就会关闭。 – 呼呼安 2012-07-28
0
@呼呼安 在 socket服务端程序中 开头增加 ignore_user_abort(1); set_time_limit(0); 这两句,就可以运行一次关浏览器了。
另外还可以在命令行运行,如 /usr/local/php/bin xxx.php & – 冯义军 2012-07-28
0
@冯义军 set_time_limit(0)这个我加了, ignore_user_abort(1);这个没加就不行吗 – 呼呼安 2012-07-30
0
@呼呼安 ignore_user_abort();主要是设置与客户机断开是否会终止脚本的行,set_time_limit() 主要是指定程序运行的最大时间。 – 冯义军 2012-08-02
显示更多隐藏的评论
呼呼安
2 票
呼呼安70
我貌似自己想到了一个办法。。

process()的地方修改一下, 直接写到数据库里(或着谁有更快的方法请告知)。
就是我所说的“动作队列”。

再写一个类似的php文件, 也是死循环。
while(true) {
// 从数据库里取出一条待处理动作
// 处理它
// 从数据库里删除该动作
}

甚至可以多运行几次这个文件, 就是不知道想停的时候怎么停 >_<

评论 (0) • 链接 • 2012-07-25
龙虾猫
0 票
龙虾猫1
风大做了一个基于pcntl的多进程Socket服务框架,可以参考使用下 http://code.google.com/p/mpass/[/code]
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: