在php socket编程中,因为tcp通信和web服务通信上的一定差别,就会出现标题中常见的几个问题:只能发送一次,不能主动下发,发送完就断开,socket长连接问题,与socket阻塞。
首先我们贴一段完整的接收发送消息的socket php长连接代码:
<?php
set_time_limit(0);
ob_implicit_flush();
//创建服务端的socket套接流,net协议为IPv4,protocol协议为TCP
$socket = socket_create(AF_INET,SOCK_STREAM,SOL_TCP);
socket_set_option($socket,SOL_SOCKET,SO_RCVTIMEO,array("sec"=>1, "usec"=>0 ) ); //设置接收超时1秒
$port = '8088';
/*绑定接收的套接流主机和端口,与客户端相对应*/
if(socket_bind($socket,'127.0.0.1',$port) == false){
echo 'server bind fail:'.socket_strerror(socket_last_error());
/*这里的127.0.0.1是在本地主机测试,你如果有多台电脑,可以写IP地址*/
}
//监听套接流
if(socket_listen($socket,10)==false){
echo 'server listen fail:'.socket_strerror(socket_last_error());
}
//socket_set_nonblock($socket); //非阻塞模式运行
//让服务器无限获取客户端传过来的信息
do{
if (($msgsock = socket_accept($socket)) < 0)
{
echo "socket_accept() failed: reason: " . socket_strerror($msgsock) . "\n";
break;
} else
{
$msg = "\nPHP Test Server. \n" ."用quit,shutdown,sun...等命令测试.\n";
socket_write($msgsock, $msg, strlen($msg));
socket_last_error();
$i=1;
do{
$talkback = "i=:" . $i;
$i++;
socket_write($msgsock, $talkback, strlen($talkback));
if(false ===($buf = socket_read($msgsock,10)))
{
$msg = "read nothing,send what you want--->".$i;
socket_write($msgsock, $msg, strlen($msg));
}
if (!$buf = trim($buf))
{
$talkback = "--->continue....----<" ;
socket_write($msgsock, $talkback, strlen($talkback));
continue;
}
if ($buf == 'quit')
{
break;
}
if ($buf == 'shutdown') {
socket_close($msgsock);
break 2;
}
if ($buf == 'hi')
{
echo'what are you doing?';
}
}while(true);
socket_close($msgsock);
}
} while(true);
socket_close($socket);
?>
然后我们针对上面的问题一一进行分析说明:
1. 为什么我的socket server程序发送完就会断开?我的socket server为什么只能发送一次
这个就要从建立连接后说起,第二层 do while循环就是在连接后等待处理接收client发送消息使用的,一直循环等待消息,这样就不会发送完,close掉消息 msgsock。
2. 通过1的说明,也相当于建立了socket长连接,那么就需要在代码的最开始 set_time_limit(0); 保证php程序一直在运行。
3.我的socket server为什么不能主动下发,只能在有client上行消息的时候才能跟着下发?
这个问题,就是php的socket_read函数的读取造成的,就是我们常说的消息阻塞,即使有第二层 do while循环,整个server程序也会等待在 socket_read这里,不往下执行,也不进行循环。
那么解决的方案,就是让接收消息的超时时间足够合适,比如1秒,那么在create创建socket server连接的时候就可以设置接收信息超时时间。
socket_set_option($socket,SOL_SOCKET,SO_RCVTIMEO,array("sec"=>1, "usec"=>0 ) ); //设置接收超时1秒
这样的话,在第二次 do while循环体里面,就可以根据自己的需要,随时下发消息内容给client了。
希望上面的信息对大家有帮助。
这个程序能解决大部分用到socket server场景,可以尝试。