PHP socket跨机器丢失数据问题

发现socket跨机器访问时,客户端读不到完整数据,下面是简化了一些流程的关键代码

服务端代码

<?php


$uri = 'tcp://0.0.0.0:8522';
$context = stream_context_create([]);
$errno = 0;
$errStr = '';
$server = stream_socket_server($uri, $errno, $errStr, STREAM_SERVER_BIND | STREAM_SERVER_LISTEN, $context);
$socket = socket_import_stream($server);
socket_set_option($socket, SOL_SOCKET, SO_KEEPALIVE, 1);
socket_set_option($socket, SOL_TCP, TCP_NODELAY, 1);

while ($conn = stream_socket_accept($server, 10000)) {
    stream_set_blocking($conn, 0);

    $data = [];
    $count = 0;
    while ($count < 8000) {
        $data[] = [
            'id'   => 'qwertyuiop123456',
            'name' => 'abcdef123的撒第五期红地球完活i都会去的',
        ];
        $count++;
    }

    $data = json_encode($data);
    $length = strlen($data);
    echo $length . chr(10);
    fwrite($conn, $data);
    fclose($conn);
}
fclose($server);

客户端

<?php

$uri = 'tcp://0.0.0.0:8522';
$errno = 0;
$errStr = '';
$stream = stream_socket_client($uri, $errno, $errStr, 30, STREAM_CLIENT_CONNECT | STREAM_CLIENT_PERSISTENT, stream_context_create([]));
$socket = socket_import_stream($stream);
socket_set_option($socket, SOL_SOCKET, SO_KEEPALIVE, 1);
socket_set_option($socket, SOL_TCP, TCP_NODELAY, 1);


$read[] = $stream;
$write = $e = [];
$response = '';
while (stream_select($read, $write, $e, 1)) {
    $response .= fread($stream, 1040001);
    if (strlen($response) >= 1040001) {
        break;
    }
}
echo $response . chr(10);
echo strlen($response) . chr(10);

这个问题需要服务端访问数据大才能复现,同一台机器上客户端可以收到完整报文,但是不同机器时就收不到了

后来我把服务端的stream_set_blocking设置为 1,客户端就可以收到完整报文,非阻塞下收不到完整报文,请问是哪里有问题?

阅读 2.5k
1 个回答

stream_set_blocking引起的问题。
由于非阻塞模式,fwrite的数据有可能写入不完整就返回了(阻塞模式不会,一定是写完才返回,写不完就阻塞直到写完),所以客户端收到的数据被截断了。

系统的I/O是有速度限制的,比如100KB/s,超过该限制的话需要等待网卡就绪才能写入。

  • 阻塞模式下,fwrite一直等待,直到系统底层可写才发送,数据最终也会被完整写入。
  • 非阻塞模式下, fwrite会返回当前写入成功的字节数,如果写入成功的字节数小于你传递的数据长度,则证明可写字节不够,需要调用stream_select对可写流进行可写状态监听,然后把剩余的数据进行fwrite
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
宣传栏