Unix套接字发送接收客户端到服务器大块数据错误

时间:2014-11-22 14:42:51

标签: php c++ sockets

我遇到了client(php)-server(c++)与大量数据通信的问题。 Server(c ++)无法从客户端(php)接收大块数据。当客户端发送少量信息时,一切正常,但是当我尝试发送大块(即~8kb)时,我断开了通信(不知道服务器收到了什么,因为它是后台进程)。

请接收我使用的以下代码并在可能的情况下予以纠正:

连接后的PHP部分(发送$sendbuf):

        socket_set_nonblock($fd);               
        $b_time = 20;
        $b_tries= 0;
        while($b_time)
        {
            if (($sc=@socket_connect($fd, $file))==FALSE && $b_tries > $b_time)
            {
                @socket_close($fd);
                return false;
            }
            else if ($sc == true)
            {   
                break;
            }
            $b_tries++;
        }

        $b_time=time();
        while($b_time)
        {
            $wd=array();

            array_push($wd,$fd); 

            if (false===($num = @socket_select($rd=NULL, $wd, $er=NULL, $select_timeout)))
            {
                @socket_close($fd);
                return $result;
            }
            else
            {
                if ($num > 0)
                {
                    if (false===($out_now=@socket_write($fd, $sendbuf, strlen($sendbuf))))
                    {
                        @socket_close($fd);
                        return false;
                    }

                    $b_time=$tm;

                    $sendbuf=substr($sendbuf,$out_now);

                    if (!strlen($sendbuf)) break;
                }
                else if ((time()-$b_time)>$timeout_write)
                {
                    @socket_close($fd);
                    return false;
                }
            }
        }

//然后从服务器接收答案,如果没问题,但没有答案。

C ++ PART:

int Socket::receive ( std::string& s ) const
{

if ( current_socket < 0 ) return false; // error socket failed!

char buf [1025];
memset(buf, 0, 1025);

// cleanup string
s = "";

int done = 0;
int status_or_nbytes = 0;
char* pch;  
do {
    status_or_nbytes = ::recv ( current_socket, buf, 1024, 0 );
    if ( status_or_nbytes == 1024 )
    {   // not all message readed, instead last byte is not SPECIAL_CHAR
        if ( buf[1023] == 26 )
        {
            //std::cout << "S 26! " << std::endl;
            buf[1023] = '\0';
            done = 1;
        }

        s += buf;
    }
    else if ( status_or_nbytes > 0 ) // < 1024
    {
            if ( buf[status_or_nbytes-1] == 26 )
            {
                buf[status_or_nbytes-1] = '\0';
                done = 1;
            }

            s += buf;
    }
    if (status_or_nbytes <= 0) 
    {
        done = 1;
    }

} while (!done);


if ( status_or_nbytes == -1 )
{
    std::cout << "ERROR1" << std::endl;
    return 0;
}
else
{
    return 1;
}
}

1 个答案:

答案 0 :(得分:2)

您收到的C代码似乎做出以下假设。

  • 数据以1024或更少的块
  • 到达
  • 每个块都被神奇的26
  • 终止
  • 如果块小于1024
  • ,则发生通信结束

但这并非必须如此。即使流中仍有大量数据等待您,操作系统也可以在recv(...)呼叫中为您提供尽可能少的数据。这更容易变得越大。

一旦收到少于1024字节,服务器就会关闭连接,PHP脚本会看到写入错误。