- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
编辑: *测试工具中的错误导致我误解了结果。 socket_select()
的工作方式完全如您所料:它确实会等待,直到套接字上的数据准备就绪。但是如果你在客户端关闭连接后调用它,它会报告就绪。原来这都是我的错,但我会把问题留在这里以防其他人怀疑 socket_select()
的行为。
我有一个多线程 PHP 应用程序,我使用套接字在线程之间进行通信。通信工作得很好,但我最终做了很多没有准备好数据的套接字的不必要读取。也许我缺少有关套接字编程的某些东西。
PHP doc对于 socket_select()
表示 read
数组中列出的套接字将被监视以查看读取是否不会阻塞。
这正是我想要的行为:调用 socket_select()
等待我的子线程之一尝试与我交谈。但这只适用于线程第一次写入,在我接受连接之后。之后,socket_select()
将永远说套接字已准备就绪,即使我已经从中读取了所有数据。
有没有什么方法可以将该套接字标记为“未就绪”,以便 socket_select()
在更多数据到达之前不会报告它已就绪?还是在我读取所有数据后关闭该连接并等待另一个连接请求是常规的?我已经阅读了很多教程和解释,但一直无法弄清楚如何正确地做到这一点。也许我遗漏了一些明显的东西?
如果它有助于查看代码,这就是我正在做的:
// Server side setup in the main thread
$this->mConnectionSocket = socket_create(AF_INET, SOCK_STREAM, 0);
$arrOpt = array('l_onoff' => 1, 'l_linger' => 0);
@socket_set_option($this->mConnectionSocket, SOL_SOCKET, SO_LINGER, $arrOpt);
@socket_set_option($this->mConnectionSocket, SOL_SOCKET, SO_REUSEADDR, true);
@socket_bind($this->mConnectionSocket, Hostname, $this->mPortNumber);
@socket_listen($this->mConnectionSocket);
@socket_set_block($this->mConnectionSocket);
.
.
.
// Then call listen(), which looks like this:
public function listen(&$outReadySockets) {
$null = null;
while(true) {
$readyArray = array_merge(array($this->mConnectionSocket), $this->mReceiverSockets);
socket_select($readyArray, $null, $null, $waitTime = null);
if(in_array($this->mConnectionSocket, $readyArray) === true) {
$this->acceptConnection();
$key = array_search($this->mConnectionSocket, $readyArray);
if($key === false) {
throw new IPCException("array_search() returned unexpected value");
} else {
unset($readyArray[$key]);
if(in_array($this->mConnectionSocket, $readyArray) === true) {
throw new IPCException("in_array() says the key is still there");
}
}
}
if(count($readyArray) > 0) {
$outReadySockets = array_merge($readyArray);
break;
}
}
}
// Client side setup in the child thread
$this->mSocket = @socket_create(AF_INET, SOCK_STREAM, 0);
@socket_set_block($this->mSocket);
@socket_connect($this->mSocket, Hostname, $this->mPortNumber);
.
.
.
@socket_write($this->mSocket, $inDataToWrite, $lengthToWrite);
// Main thread reads the socket until it's empty
$data = "";
$totalBytesRead = 0;
while($totalBytesRead < $inNumberOfBytesToRead) {
// Strange that even if we set the socket to block mode, socket_read()
// will not block. If there's nothing there, it will just return an
// empty string. This is documented in the PHP docs.
$tdata = socket_read($inSock, $inNumberOfBytesToRead);
if($tdata === false) {
throw new IPCException("socket_read() failed: " . socket_strerror(socket_last_error()));
} else {
$data .= $tdata;
$bytesReadThisPass = strlen($tdata);
if($bytesReadThisPass === 0) {
break;
}
}
$totalBytesRead += $bytesReadThisPass;
}
.
.
.
// Then calls listen() again
正如我所说,它工作得很好,除了当我第二次调用 listen()
时,它告诉我套接字仍然准备就绪。这似乎是 PHP 文档所说的,但我不想要那样。我想知道什么时候那里真的有数据。我做错了吗?或者只是错过了重点?
最佳答案
你滥用了套接字函数。
首先,socket_read
会在出错时返回false
;您的代码不会对此进行检查,并将错误返回视为与空字符串相同(这是您正在使用的特定构造的产物,即字符串连接和 strlen
)。
另一个问题是,当检测到连接尝试时,您显然违反了 socket_select
的说明:
No socket resource must be added to any set if you do not intend to check its result after the socket_select() call, and respond appropriately. After socket_select() returns, all socket resources in all arrays must be checked. Any socket resource that is available for writing must be written to, and any socket resource available for reading must be read from.
取而代之的是,如果检测到连接,代码会接受它并再次返回到 socket_select
;准备好读取的套接字未得到服务。
最后,您似乎对 EOF 在套接字上的含义感到困惑:这意味着客户端已关闭连接的写入端。一旦 socket_read
第一次返回空字符串(不是 false
!),它将永远不会再返回任何有意义的内容。一旦发生这种情况,您可以从服务器的写入端发送数据和/或完全关闭连接。
关于PHP:如何正确使用 socket_select() 和 socket_read(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18434695/
试图让我的socket_read()在之前已经读取过数据时阻塞。 我需要它等待$ spawn套接字具有新数据(可以相等,即希望它在接收到“FE”时读取两次),然后再次发送。 当没有任何新数据时,如何阻
在 linux 中使用 socket_read() 时,我遇到了一个非常奇怪的行为。 我正在使用缓冲区限制为 2048 的 socket_read。 虽然在我的 Windows 系统上它获得了整个响应
我使用下面的脚本连接到外部PBX服务器并获取调用日志。 但是,它在等于1 log的时间仅返回1刺。因此,我需要刷新多次以获取所有可用的日志。 $socket = socket_create(AF_IN
我最近开始在 PHP 上练习套接字,但遇到了一个我找不到文档的问题。我在 C++ 中看到过类似的情况,但对此没有明确的答案。编码: do { $input = socket_read($cli
我在运行 $data = socket_read($socket, 2048, PHP_BINARY_READ); 应该返回这样的字符串 // $data == ' 1840 kg G 008765
从阅读PHP手册来看,socket_recv和socket_read函数看起来和我一样,这两个函数都从客户端获取数据。 谁能告诉我这两个函数有什么不同? 最佳答案 socket_recv 返回接收到的
我尝试使用 PHP 套接字创建从 C# 应用程序到 PHP 5.3 脚本的 tcp/ip 套接字连接。 C# 应用程序应将 JSON 字符串发送到 PHP 脚本。 我关于 the socket_rea
我想知道如何在 socket_read 调用上设置超时?第一次调用 socket_read 时,它会等待数据发送,如果 5 秒内没有数据发送,我想关闭连接。有帮助吗?我已经尝试过 SO_RCVTIME
我正在学习套接字编程,并尝试使用php。 我想使用客户端连接到套接字服务器,并从客户端读取服务器的响应。 的代码: Server.php: $address="127.0.0.1"; $port=33
我正在尝试从我在 Java 应用程序上编写的套接字读取响应。然而 PHP 只是挂起。这是我用来从套接字读取的代码: while($resp = socket_read($socket, 1000))
编辑: *测试工具中的错误导致我误解了结果。 socket_select() 的工作方式完全如您所料:它确实会等待,直到套接字上的数据准备就绪。但是如果你在客户端关闭连接后调用它,它会报告就绪。原来这
编辑:我已经解决了自己的问题并修复了下面的代码以反射(reflect)这些更改,以防有人遇到同样的困难。此代码适用于 PHP 客户端和 java 服务器连接。 我知道论坛上充斥着这些问题...但是,在
我正在编写一个 PHP 服务器,客户端以特定的字符编码发送数据。现在,我想告诉服务器以相同的字符编码集读写数据。 我应该如何为PHP的socket_read指定字符编码集?和 socket_write
我正在尝试创建一个 PHP 客户端来连接到一个简单的 Java 服务器,发送一些数据,然后接收返回的响应。 Java 服务器在与用 Java 编码的简单客户端交互时响应良好,但是一旦从 PHP 客户端
我用 socket_create()创建套接字资源,然后绑定(bind) IP通过 socket_bind() 向它发送地址,它的工作正常; 但过了一会儿(超过 30 分钟)在行 socket_rea
我是一名优秀的程序员,十分优秀!