PHP套接字服务器,检查客户端是否处于活动状态

问题描述 投票:0回答:5

我有一个 php 服务器正在监听 1 个 c# 客户端。

建立连接后,它会保持活动状态,直到客户端发送命令“quit”,该命令会终止 PHP 服务器。

但是,当 C# 客户端在没有“退出”命令的情况下断开连接时(即:单击 Windows 窗体中的关闭 (x) 按钮),服务器只会继续侦听,并且无法从该客户端接收任何其他连接。

有没有办法从服务器端(PHP)检查与客户端的连接是否仍然有效?

我的php服务器代码基于以下示例1:http://php.net/manual/en/sockets.examples.php

如果有人有兴趣重现错误/错误行为,请粘贴示例1中的代码:http://php.net/manual/en/sockets.examples.php,从局域网中的远程客户端通过telnet连接,拔掉客户端线... php 服务器将永远挂起,不接受新连接。

php sockets listener keep-alive
5个回答
2
投票

在循环中,您需要检查

socket_read()
的返回值。如果它返回 FALSE,则存在读取错误(这可能是由远程主机关闭连接引起的)。您提供的链接中的示例代码涵盖了这种情况。

如果您需要优雅地处理某些错误状态,您始终可以使用

socket_last_error()
检查套接字错误代码 -- 此注释 描述了可能的代码。

编辑:

当使用 putty 进行 telnet 时,如果我用 X 按钮关闭,则 PHP 中的连接会正确关闭,但如果我拔掉 putty 机器的以太网线,PHP 服务器就会挂起。

杀死PuTTY时连接被关闭的原因是PuTTY在退出时关闭了其打开的连接。这会导致

socket_read()
返回错误代码(我相信
ECONNRESET
)。如果你拔掉网线,它就没有机会这样做。

根据您的网络配置方式,TCP 连接应该最终会失败。您可以尝试通过设置

SO_RCVTIMEO
socket_set_option()
来控制超时,但这并不总是适用于所有平台(我正在看着你,WinSock)

或者,您可以使用

socket_select()
以及合理的超时来滚动自己的轮询循环。如果超时后所有连接的套接字都没有数据要发送,则终止服务器。


0
投票

我通过检查

socket_select()
$read
数组是否包含有问题的连接来完成此操作,但
socket_read()
数据为空(两次)。

看起来

socket_select()
将断开连接的客户端添加到
$read
数组中,并且
socket_read()
在尝试读取断开连接的客户端数据时给出一个空字符串
''


0
投票

以下代码将显示连接状态。

  $address='example.com';
    $port = '9065';
    if (isset($port) && ($socket=socket_create(AF_INET, SOCK_STREAM, SOL_TCP))
     && (socket_connect($socket, $address, $port)))  {
    $text="Connection successful on IP $address, port $port";
    socket_close($socket);
    }
    else  {
    $text='Unable to connect<pre>'.socket_strerror(socket_last_error()).'</pre>';
    }
    echo $text;

0
投票

正如 Michael Dodd 所说,使用 socket_select() 可以拥有所有已接收数据的套接字以及它们已关闭的套接字。此外,您还可以拥有其缓冲区可以接受数据传输的所有那些,和/或那些引发一些异常的数据。确实有此信息,套接字数组的(单独)副本必须放置在函数的第二个和/第三个参数中。 (如果不需要,此函数必须将 $null 作为等于 null 的变量,而不仅仅是 null)

以下示例显示如何读取数据并测试套接字是否仍处于打开状态。我将此代码用于侦听套接字上由 socket_accept() 创建的套接字,并且它工作正常。可以使用socket_read代替socket_recv。

//all sockets created have been inserted to an array $socketArray
//get a copy of the sockets array
$tempArray = $socketArray;
//this command will remove from $tempArray all sockets that have no data and are alive, with a timeout of 0 sec, 100 msec
socket_select($tempArray, $null, $null, 0, 100);
if (count($tempArray)) {
    //if we have some sockets in the array
    foreach($tempArray as $socket) {
        //read some data
        $count = socket_recv($socket, $socketData, 1024, 0);
        if ($count) {
            //your code to do what you want with $socketData
        } else {
            //find socket position in initial socket array
            $index = array_search($socket, $socketArray);
            //if found, remove it from array and close the socket
            if ($index !== false) {
                array_splice($socketArray, $index, 1);
                socket_close($socket);
            }
        }
    }
}

0
投票

is_resource()函数可用于检查socket_accept()创建的套接字的连接状态。

© www.soinside.com 2019 - 2024. All rights reserved.