5

我在实现一个简单的 TCP 服务器时遇到了困难。以下代码取自boost::asio 示例,准确地说是“Http Server 1”。

void connection::start() {
    socket_.async_read_some(
            boost::asio::buffer(buffer_),
            boost::bind(
                &connection::handle_read, shared_from_this(),
                boost::asio::placeholders::error,
                boost::asio::placeholders::bytes_transferred
            )
        );
}
void connection::handle_read(const boost::system::error_code& e, std::size_t bytes_transferred) {
        if (!e && bytes_transferred)    {
                std::cout << " " << bytes_transferred <<"b" << std::endl;
                data_.append(buffer_.data(), buffer_.data()+bytes_transferred);

                //(1) what here?                
                socket_.async_read_some(
                    boost::asio::buffer(buffer_), 
                    boost::bind(
                        &connection::handle_read, shared_from_this(),
                        boost::asio::placeholders::error,
                        boost::asio::placeholders::bytes_transferred
                    )
                ); 

            }
            else// if (e != boost::asio::error::operation_aborted)
            {
                std::cout << data_ << std::endl;
                connection_manager_.stop(shared_from_this());
            }
        }

在原始代码中,buffer_足够大以保留整个请求。这不是我需要的。我已将大小更改为 32 字节。

服务器编译并监听 localhost 的 80 端口,所以我尝试通过我的网络浏览器连接到它。

现在如果语句 (1) 被注释掉,那么只有请求的前 32 个字节被读取并且连接挂起。网络浏览器一直在等待响应,服务器确实......我不知道是什么。

如果 (1) 未注释,则读取整个请求(并添加到data_),但它永远不会停止 - 我必须在浏览器中取消请求,然后该else { }部分才会运行 - 我在标准输出上看到我的请求。

问题1:我应该如何处理一个大的请求?
问题2:我应该如何缓存请求(目前我将缓冲区附加到一个字符串)?
问题3:如何判断请求已经结束?在 HTTP 中总是有响应,所以我的网络浏览器一直在等待它并且没有关闭连接,但是我的服务器如何知道请求已经结束(可能关闭它或回复一些“200 OK”)?

4

1 回答 1

9

假设浏览器向您发送 1360 字节的数据,您说asio要将一些数据读入缓冲区,但您说它只有 32 字节。然后第一次调用它时,您的处理程序将以 32 字节的数据开头调用。在这里,如果您评论 (1),那么浏览器会尝试发送其其余数据(实际上浏览器已经发送了它,并且它在操作系统缓冲区中等待您从那里窥视它),您可能会io_service::run因为一些奇迹而被阻止!

如果您在说循环开始时取消注释(1),则读取第一个块,然后读取下一个块,然后...直到浏览器发送的数据完成,但是在那之后,当您说asio要读取更多数据时,它将等待一些永远不会来自浏览器的数据(因为浏览器已经发送了它的信息并且正在等待你的回答),当你取消来自浏览器的请求时,它将关闭它的套接字,然后你的处理程序将被调用,并显示一个错误说我无法读取更多数据,因为连接已关闭。!!

但是你应该在这里做的是:你应该学习HTTP格式,从而知道你的浏览器发送给你的数据是什么,并为它提供一个好的答案,然后你与客户端的通信将继续进行。在这种情况下,缓冲区的结尾是\r\n\r\n,当您看到它时,您不应再读取任何数据,您应该处理到目前为止所读取的内容,然后向浏览器发送响应。

于 2012-09-16T21:26:27.577 回答