std::string内部空间地址使用

文章介绍了如何使用std::string高效地进行Base64编码,避免频繁内存申请和拷贝。提到在不同编译器环境下可能存在的问题及解决方案。

     std::string提供两种方法获取内部空间地址:c_str()和data()。不幸的是返回值全是const char*类型。即使外部强制转换为char*类型也只能修改对应的字符,如果想使用std::string::data()作为参数去保存不定长度的字符串时,std::string::size无法实时修改;如果直接使用外部字符串直接对std::string赋值,又避免不了内存申请和拷贝。

     为了提高效率,减少内存申请和拷贝,使用如下方式可高效构建字符串

int PictureEncoderToBase64(std::string& dest, const uint8_t* picture, int pictureLen, const std::string& pictureType) {
    std::string prefix = "data:image/" + pictureType + ";base64,";
    // 原始字节是3个 编码后的字节数为4个
    int externSize = pictureLen + (pictureLen + 3) / 3 + prefix.size() + 32;
    // 申请空间,直接填充dest指向的内存地址,不会更改dest的size
    dest.reserve(externSize);
    char* encoderBuffer = (char*)dest.data();
    if (nullptr == encoderBuffer) {
        return -1;
    }
    
    memset(encoderBuffer, 0, externSize);
    base64_encode(encoderBuffer + prefix.size(), externSize - int(prefix.size()), picture, pictureLen);
    memcpy(encoderBuffer, prefix.c_str(), prefix.size());
    // assign内部使用memmove拷贝数据,encoderBuffer和dest.data()指针相同,memmove不会复制,提升效率。
    // assign更新dest的size
    // dest.assign(encoderBuffer, strlen(encoderBuffer));
    dest.append(encoderBuffer, strlen(encoderBuffer));

    return 0;
}

     注意,上述代码在Windows上(visual studio2022)以及linux gcc 8.3.1上运行正常;但是在gcc (Debian 10.2.1-6) 10.2.1 20210110运行失败。此处将assign改为append即可

### 如何为 `std::string` 分配或设置空间 在 C++ 中,`std::string` 是标准库提供的一种动态字符串类。为了提高性能或预先分配足够的内存以减少重新分配次数,可以使用多种方法来控制其容量。 #### 使用 reserve 方法预分配空间 通过调用成员函数 `reserve(size_t n)` 可以为 `std::string` 预先分配至少 `n` 字符的空间。这不会改变当前字符串的内容长度,仅影响内部缓冲区大小: ```cpp #include <iostream> #include <string> int main() { std::string str; // Preallocate memory for at least 100 characters. str.reserve(100); std::cout << "Capacity of string is now: " << str.capacity() << '\n'; } ``` 此操作有助于当预期要向字符串追加大量数据时避免频繁的内存重分配开销[^1]。 #### 构造具有初始容量的字符串 另一种方式是在创建 `std::string` 对象时指定初始化参数,从而间接设定起始容量。例如,可以通过重复字符构造特定长度的字符串: ```cpp // Create a string initialized with 50 'A' characters. std::string largeString(50, 'A'); ``` 这种方式不仅设置了最小容量还设定了实际内容长度。如果只需要预留空间而不填充任何有效值,则应优先考虑 `reserve()` 函数[^2]。 对于多线程环境下的局部变量管理,虽然提到过 `thread_local` 关键字用于定义每线程实例化的变量,但这与单个 `std::string` 的内存分配策略无关,在这里不作深入讨论。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值