直接从C API填充std :: string

Tyl*_*wis 4 c++ c++11 c++14 c++17

我发现自己经常围绕C风格的API编写c ++包装器,我写的一些常见的浪费代码如下:

//getSomeString() wraps C api that gets some C string from somewhere
std::string MyClass::getSomeString()
{
    char buffer[BUFFER_MAX];
    memset(buffer, '\0', BUFFER_MAX);
    auto result = GetCApiString(buffer, BUFFER_MAX); //C style string getter 
    return (result == NO_ERROR) ? std::string{buffer} : ""; //Copy here
}
Run Code Online (Sandbox Code Playgroud)

但我真的想做这样的事情:

//getSomeString(): as before
std::string MyClass::getSomeString()
{
    DirectStringFillIterator<char> returnString; // <--HERE. Is something like this possible?
    auto result = GetCApiString(returnString, BUFFER_MAX);
    return (result == NO_ERROR) ? returnString.str() : ""; 
}
Run Code Online (Sandbox Code Playgroud)

这样的事情可能吗?

Mat*_*lia 9

std::string从C++ 11开始就保证是连续的(但在我遇到的每个实现中,甚至在C++ 03中也是这样).所以,通常你可以做到

std::string MyClass::getSomeString() {
    std::string ret(BUFFER_MAX, '\0');
    if(GetCApiString(&ret[0], ret.size()) != NO_ERROR) return "";
    ret.resize(ret.find_first_of(0));
    return ret;
}
Run Code Online (Sandbox Code Playgroud)

这样可以避免复制第一个方法(从本地缓冲区到返回的字符串),但会强制返回的字符串BUFFER_MAX始终分配字节,这可能是您的情况下的缺点,也可能不是.

  • @MatteoItalia将在C++ 17中添加一个新的非`constst`重载(对于`.data()`,而不是`.c_str()`). (3认同)