首先,我想声明我是C ++的初学者并且总体上有所提升,尽管我对Java和C#等语言有一定的经验。
话虽这么说,我在通过Boost TCP套接字传输数据时遇到了问题。我正在使用此代码:
发送:
tcp::socket s(io_service);
boost::asio::connect(s, iterator);
string response = static_cast<ostringstream*>( &(ostringstream() << arg1) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg2) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg3) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg4) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg5) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg6) )->str();
//pad the rest
while (response.length() < max_length)
{
response += ' ';
}
boost::asio::write(s, boost::asio::buffer(response, sizeof(response)));
阅读:
tcp::acceptor a(io_service, tcp::endpoint(tcp::v4(), port));
while(true)
{
socket_ptr sock(new tcp::socket(io_service));
a.accept(*sock);
boost::thread t(boost::bind(session, sock));
}
...
void session(socket_ptr sock)
{
char data[max_length];
size_t length;
try
{
while(true)
{
boost::system::error_code error;
//size_t length = sock->read_some(boost::asio::buffer(data), error);
length = boost::asio::read(*sock, boost::asio::buffer(data, sizeof data));
std::cout << "Length " << length << endl;
if (error == boost::asio::error::eof)
{
break; // Connection closed cleanly by peer.
}
else if (error)
{
throw boost::system::system_error(error); // Some other error.
}
}
}
catch (std::exception& e)
{
std::cout << data << endl;
string message = e.what();
if(std::string::npos != message.find(END_OF_FILE))
{
domeSomethingWithTheData(data);
memset(data, 0, sizeof data);
}
else
{
std::cout << data <<endl;
std::cerr << "Exception in thread: " << e.what() << "\n";
}
}
}
我无法理解的是,无论我使用什么(boost::asio::read
或sock->read_some
),我的数据都不会超过32个字符。我已经尝试将缓冲区的大小增加到2048
,但问题仍然存在。
缓冲区总是包含超过32个字符(其中大部分是垃圾)但我假设这些字符实际上并非从客户端发送,而是来自服务器端缓冲区的实际分配,因为{{1打印出32。
有关我如何能够从客户端向服务器发送任意数量的字符的任何信息将不胜感激。
编辑:
在Rafal Rawicki提供的指针之后,我现在正在使用此代码:
std::cout << "Length " << length
客户端保持不变,唯一的例外是我最后添加 boost::asio::streambuf buffer;
boost::asio::read_until(*sock, buffer, '\n');
std::istream is(&buffer);
std::string line;
std::getline(is, line);
std::cout << line << std::endl;
。这产生了很多\n
例外。
答案 0 :(得分:3)
这实际上不是你问题的答案。但快速提出建议。
你的字符串连接代码很残忍!
string response = static_cast<ostringstream*>( &(ostringstream() << arg1) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg2) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg3) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg4) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg5) )->str()
+ "," + static_cast<ostringstream*>( &(ostringstream() << arg6) )->str();
对于每个arg,您创建一个新的ostringstream
,将值写入其中,然后不必要地取消引用它,然后不必要地将其强制转换为该类型无论如何它已经有了。最后,您使用字符串连接来构建最终字符串。
这种字符串连接既不高效,也不复杂(需要临时对象,一些浪费的分配等)。
ostringstream
是字符串连接和格式化的正确工具,它高效而复杂。
长话短说,你的代码应该是这样的:
ostringstream oss;
oss << arg1 << "," << arg2 << "," << arg3 << "," << arg4 << "," << arg5 << "," << arg6;
string response = oss.str();
答案 1 :(得分:2)
boost::asio::read
应该读取数据,直到缓冲区已满或发生错误。
boost::asio::socket::read_some
应该从套接字中读取某些数据。
要阅读所需数量的数据,您可以:
read_until
将数据读取到预期的分隔符boost::asio::read
重载,它以参数completion_condition
为例,例如,字节数。另外,请查看带有Boost.Asio的issues with synchronous operations并考虑使用异步操作。
编辑:我错过了客户端代码中不正确的缓冲区初始化。使用boost::asio::read
是正确的。
您不能像这样初始化缓冲区:
boost::asio::buffer(response, sizeof(response));
因为sizeof
返回std::string
结构的大小,而不是整个字符串。在我的机器上它始终是8.使用:
boost::asio::buffer(response);