我正在用流协议(protocol)编写服务器,所以我需要做一些事情,比如找到 header 的结尾,复制它,然后解析 boost 缓冲区中的其他内容。正如我发现使用字符串进行操作(在其中查找字符串、使用迭代器复制/删除等)的最佳方式是 std::string。但我正在使用 char 数组缓冲区。所以我需要有两个缓冲区 - char 数组和 std::string - 每次我需要使用缓冲区进行操作时,我都需要将 char 数组转换为 std::string,执行我的操作然后将其转换回来使用 std::string.c_str() 转换为 char 数组。我发现的另一种方法是使用 streambuf(如我在之前的 questition 中所问),然后为其创建 istream/ostream 并将其中的内容填充到 std::string(如 documentation 中所示)。
对于 streambuf,我需要:
流缓冲区
可变缓冲区类型
流
操作系统
和 std::string
但是使用 char 数组和 std::string 我只需要:
字符数组
标准::字符串
所以我认为使用 streambuf 是浪费内存(我需要为每个连接创建缓冲区)。我可以使用 std::string 作为 boost 缓冲区吗?但是我认为可能有更好的方法来做到这一点。你能给我一个建议吗?
编辑:
我需要用我的缓冲区做这样的事情,但 char 数组不提供像 std::string (erase, substr, ...) 这样的功能,所以我需要使用 std::string 作为缓冲区。将它用作 boost::buffer 的最佳方式是什么,或者像这段代码一样进行解析的最佳方式是什么?
#include <iostream>
int main(int argc, char* argv[])
{
//"header" is header
//"end" is marking that this point is end of header
//"data" is data after header
//this all is sent in one packet which I receive to buffer
//I need to fill "headerend" to std::string header and then remove "headerend" from begining of buffer
//then continue parsing "data" which stay in buffer
std::string buffer = "headerenddata"; //I receive something like this
std::string header; //here I'll fill header (including mark of end of header)
//find end of header and include also mark of end of header which is "end" (+3)
int endOfHeader = int(buffer.find("end"))+3;
//fill header from buffer to string header
header = buffer.substr(0, endOfHeader);
//delete header from input buffer and keep data in it for next parsing
buffer.erase(buffer.begin(), buffer.begin()+endOfHeader);
//will be just "data" becouse header and mark of header are removed
std::cout << buffer << std::endl;
//will be "headerend" which is "header" and mark end of header which is "end"
std::cout << header << std::endl;
return 0;
}
最佳答案
您可以使用 std::string
作为构造 boost::asio::buffer
的参数.不过,我通常使用字符 vector ( std::vector<char>
)。我认为处理它更容易,但这可能实际上取决于您其余代码的设计方式。
您也可以使用 C++11 array .它的行为类似于 vector ,但它静态分配空间(即,一旦创建数组,底层缓冲区的大小就不能更改)。在某些情况下,这可能会给您带来一些性能优势。如果您不能使用 C++11,Boost 也包含一个非常相似的类。
boost::asio::buffer
也接受普通的字符数组 ( char buf[SIZE]
),但如果可能的话,使用前面提到的选项可能更方便。
作为引用,这里是 boost::asio::buffer
的文档.
更新:为了避免从char[]
的转换至 string
, 你可以使用 vector
用于接收和处理缓冲区(使用 string
接收缓冲区也可能有效,但我从未尝试过)。
所以,为了接收你可以这样做:
vector<char> msg(N);
...
asio::read(socket, asio::buffer(msg), asio::transfer_at_least(N), ec);
然后为了处理数据包并拆分 header 和数据,您可以使用迭代器,避免代码中复杂度为 O(n) 的操作(substr
和 erase
)。当然find
(或在我的示例中为 search
)无法避免:
string end_marker = "end";
auto it = search(msg.begin(), msg.end(), end_marker.begin(), end_marker.end());
process_header(msg.begin(), it + 2);
process_data(it + 3, msg.end());
关于c++ - 使用 boost 缓冲区进行操作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10775957/