c++ - 使用 boost 缓冲区进行操作

标签 c++ boost buffer stdstring

我正在用流协议(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) 的操作(substrerase)。当然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/

相关文章:

c++ - 随机访问缓冲区优化

Java 调整双缓冲区大小

c - 读取 TCP 套接字中的 buff

c++ - BOOST_FOREACH : is there a trick to avoid the all-caps spelling?

c++ - 链接 boost 库

c++ - 如何在 CppUnit 中漂亮地打印 STL 数据结构?

c++ - 成员函数指针的奇怪 C++ 规则?

c++ - 当值部分不存在时使用 Boost Property Tree 读取 INI 文件

c++ - DirectX CreateOverlay函数库?

c++ - Valgrind:是否可以接受越来越多的可能丢失的字节?