0

我需要一些帮助来编写一个 http 客户端。当我尝试从网络服务器接收数据时,问题就来了。recv() 调用会阻塞程序。任何更好的方向都会非常有帮助,我将在下面发布我的代码:

if ( argc != 2 )
{
    cerr << "Usage: " << argv[0];
    cerr << " <URI>" << endl;
    return 1;
}
else
{
    uri_string = argv[1];
}

// Create URI object and have it parse the uri_string
URI *uri = URI::Parse(uri_string);

if ( uri == NULL )
{
    cerr << "Error: Cannot parse URI." << endl;
    return 2;
}

// Check the port number specified, if none use port 80
unsigned port = 80;
if ( uri->Is_port_defined() )
{
    port = uri->Get_port();
}

// Create TCP socket and connect to server
int tcp_sock = socket( AF_INET, SOCK_STREAM, 0 );
if ( tcp_sock < 0 )
{
    cerr << "Unable to create TCP socket." << endl;
    return 3;
}

sockaddr_in server;
socklen_t slen = sizeof(server);

server.sin_family = AF_INET;
server.sin_port = htons( port );
hostent *hostp = gethostbyname( uri->Get_host().c_str() );
memcpy( &server.sin_addr, hostp->h_addr, hostp->h_length );

if ( connect( tcp_sock, (sockaddr*)&server, slen ) < 0 )
{
    cerr << "Unable to connect to server via TCP." << endl;
    close( tcp_sock );
    return 4;
}

// Build HTTP request to send to server
HTTP_Request *request = HTTP_Request::Create_GET_request( uri->Get_path() );
request->Set_host( uri->Get_host() );
string request_string = "";
request->Print( request_string );

//cout << request_string << endl;

// Send it to the server, wait for reply and use HTTP_Response to get reply
send( tcp_sock, &request_string, sizeof(request_string), 0 );

char recv_buffer[1024];
int bytes_recv = 0;
while ( bytes_recv < 1024 )
{
    int recv_len = recv( tcp_sock, recv_buffer + bytes_recv,
        1024 - bytes_recv, 0 );
    if ( recv_len == -1 )
    {
        cerr << "Error receiving response from server." << endl;
        close( tcp_sock );
        return 5;
    }
    bytes_recv += recv_len;
}


HTTP_Response *response = HTTP_Response::Parse(recv_buffer, bytes_recv);
string response_string = "";
response->Print( response_string );
cout << response_string << endl;

return 0;

}

4

5 回答 5

3

您正在使用阻塞 TCP/IP 套接字,但您没有查看 HTTP 回复的“Content-Length”标头以了解要读取的字节数。您当前的读取逻辑是在循环中调用 recv() 直到收到最大 1024 字节。如果服务器发送的字节数少于 1024,您将无限期地被阻止,因为您调用 recv() 的次数过多,请求的字节数过多。

于 2009-10-20T20:03:46.683 回答
1

recv()应该阻止,直到它得到响应。您确定您正在正确编写您的请求,并且服务器正在响应它吗?可以将文件描述符置于非阻塞模式并使用select()or对其进行测试poll(),但我的猜测是您只是在某处存在协议错误。您期望的行为是什么?

于 2009-10-16T22:01:05.030 回答
0

这是一个问题吗?

如果客户端是命令行很好。
如果是 GUI,那么检索数据的线程应该与 UI 线程不同。

但一个解决方案是使用 select()
这将告诉你是否有任何东西要从端口读取。
从而允许您在等待时做其他工作。

于 2009-10-16T21:57:39.200 回答
0

HTTP 请求应以空行结束,即


GET / HTTP/1.1
Host: blah.com
                    <- this here is an empty line

看起来您的代码没有这样做(它可能应该说request->Print(request_string + "\n")

题外话:你知道 C 中有现成的 HTTP 客户端,对吧?(例如 libcurl)。

于 2009-10-17T06:01:15.470 回答
0

您必须接收内容长度字段中的许多字节。您必须将 line: 更改if ( recv_len == -1 ) 为:

if ( recv_len <= 0 ) break;
else if ( recv_len == -1 )

因为 0 是服务器在发送所有数据后断开连接的时间。

于 2010-01-14T10:46:05.977 回答