5

我是套接字连接的新手,我想向套接字服务器发送一些命令,但我只能发送第一条消息而不能发送其他消息。

客户:

$fp = stream_socket_client("tcp://127.0.0.1:1000", $errno, $errstr, 30);
if (!$fp) {
    echo "$errstr ($errno)<br />\n";
} else {
    $cmd_1 = "hello";
    fwrite($fp, $cmd_1);
    while (!feof($fp)) {
        print_r(fgets($fp, 1024));
    }
    $cmd_2 = "second message";
    fwrite($fp, $cmd_2);
    while (!feof($fp)) {
        print_r(fgets($fp, 1024));
    }
    fclose($fp);
}

我怎样才能像 first 一样发送多条消息$cmd_1,并且取决于结果,我必须发送$cmd_2

服务器

error_reporting(E_ALL);

/* Allow the script to hang around waiting for connections. */
set_time_limit(0);

/* Turn on implicit output flushing so we see what we're getting
 * as it comes in. */
ob_implicit_flush();

$address = '127.0.0.1';
$port = 1000;

if (($sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) === false) {
    echo "socket_create() failed: reason: " . socket_strerror(socket_last_error()) . "\n";
}

if (socket_bind($sock, $address, $port) === false) {
    echo "socket_bind() failed: reason: " . socket_strerror(socket_last_error($sock)) . "\n";
}

if (socket_listen($sock, 5) === false) {
    echo "socket_listen() failed: reason: " . socket_strerror(socket_last_error($sock)) . "\n";
}

do {
    if (($msgsock = socket_accept($sock)) === false) {
        echo "socket_accept() failed: reason: " . socket_strerror(socket_last_error($sock)) . "\n";
        break;
    }
    /* Send instructions. */
    $msg = "\nWelcome to the PHP Test Server. \n" .
        "To quit, type 'quit'. To shut down the server type 'shutdown'.\n";
    socket_write($msgsock, $msg, strlen($msg));

    do {
        if (false === ($buf = socket_read($msgsock, 2048, PHP_NORMAL_READ))) {
            echo "socket_read() failed: reason: " . socket_strerror(socket_last_error($msgsock)) . "\n";
            break 2;
        }
        if (!$buf = trim($buf)) {
            continue;
        }
        if ($buf == 'quit') {
            break;
        }
        if ($buf == 'shutdown') {
            socket_close($msgsock);
            break 2;
        }
        $talkback = "PHP: You said '$buf'.\n";
        socket_write($msgsock, $talkback, strlen($talkback));
        echo "$buf\n";
    } while (true);
    socket_close($msgsock);
} while (true);

socket_close($sock);
4

2 回答 2

1

问题在于

if (false === ($buf = socket_read($msgsock, 2048, PHP_NORMAL_READ))) {
    echo "socket_read() failed: reason: " . socket_strerror(socket_last_error($msgsock)) . "\n";
    break 2;
}

而不是将其与 false do 进行比较=== ''

if ('' === ($buf = socket_read($msgsock, 2048, PHP_NORMAL_READ))) {
    echo "socket_read() failed: reason: " . socket_strerror(socket_last_error($msgsock)) . "\n";
    break 2;
}

否则连接会过早关闭 - 请参见此处 - http://www.php.net/manual/en/function.socket-read.php#89133

于 2013-09-07T17:41:21.673 回答
1

问题出在下面一行:

while (!feof($fp)) {

这样做是:它等待该套接字上的 EOF(通常是套接字关闭)。它会为您提供第一个响应,并等待进一步的响应而不发送任何进一步的请求(while 循环)。您希望阅读您发送的每条请求消息的响应消息,对吗?

像这样修改您的代码(未经测试,但应该可以):

$cmd_1 = "hello";
fwrite($fp, $cmd_1);
print_r(fgets($fp, 1024));

$cmd_2 = "second message";
fwrite($fp, $cmd_2);
print_r(fgets($fp, 1024));

fclose($fp);

fgets(..)从套接字读取一行后将返回。你实际上不需要一个while循环。


此外,开始为您的通信定义结构/协议是一个很好的点。在客户端,您正在逐行读取,而不是服务器。读取的套接字(在服务器上)可以返回部分消息。如果您决定了通信结构,则可以将读取设计为在读取整个结构之前不返回。

于 2013-04-30T09:35:46.880 回答