wcstombs & 为堆上的字符数组分配内存

wcstombs & allocating memory for character array on heap

我正在读取一个包含单个宽字符行的文件。但是,我永远不知道它会持续多久。我已将其读入 std::wstringinString,并设法凭空创建了多字节字符串(Q1 - 这些称为 r 值吗?)。 Q2 - 现在,我如何在堆中为此分配内存并获得指向它的智能指针?我不想使用 newmalloc(并最终调用 freedelete)或任何常量将其存储在堆栈中(因为我永远不知道最大值长度)。 Q3 - 我可以在这里使用 make_sharedmake_unique 函数模板吗? Q4 - 具体来说,我可以得到一个像 shared_ptr<char> 的指针指向分配在堆上的 char 数组吗?

我试过类似下面的方法,

std::shared_ptr<char> MBString(const_cast<char*>(std::string(inString.begin(), inString.end()).c_str()));

没用。我在互联网上尝试了一些建议,但我还不知道该怎么做。

Q5 - 更不用说宽字符到多字节的转换,一般来说,我如何在堆上分配任意长度的 char 字符串并获得指向它的智能指针?

std::wfstream inFile(L"lengthUnkown.txt", std::ios::in);
std::wstring inString;
inFile >> inString;
std::wcout << inString << std::endl; //prints correctly
std::cout << (const_cast<char*>(std::string(inString.begin(), inString.end()).c_str())) << std::endl; //this prints the line correctly as expected
//convert wide character string to multi-byte on the heap pointed, to by MBString 
//std::cout << MBString << std::endl; //I want to print the multi-byte string like this
return 0;

不是资源最优但可靠:

wchar_t* mb2wstr(const char* inval) {
    size_t size = std::strlen(inval);
    #define OUTSZ (size+1)*sizeof(wchar_t)
    auto buf = (wchar_t*)std::malloc(OUTSZ);
    std::memset(buf, 0, OUTSZ);
    std::setlocale(LC_CTYPE,""); //  необходима, чтобы отработала "mbstowcs"
    size = std::mbstowcs(buf, inval, size);
    if ( size == (size_t)(-1) ) {
        std::free(buf);
        buf = nullptr;
    } else {
        buf = (wchar_t*)std::realloc(buf,OUTSZ);
    }
    return buf;
    #undef OUTSZ
}

char* wstr2mb(const wchar_t* inval) {
    size_t size = std::wcslen(inval);
    #define OUTSZ (size+1)*MB_CUR_MAX // Maximum length of a multibyte character in the current locale
    auto buf = (char*)std::malloc(OUTSZ);
    std::memset(buf, 0, OUTSZ);
    std::setlocale(LC_CTYPE,""); //  необходима, чтобы отработала "wcstombs"
    size = std::wcstombs(buf, inval, size*sizeof(wchar_t));
    if ( size == (size_t)(-1) ) {
        std::free(buf);
        buf = nullptr;
    } else {
        buf = (char*)std::realloc(buf,size+1);
    }
    return buf;
    #undef OUTSZ
}

const std::string pwchar2string(const wchar_t* inval) {
    char* tmp = wstr2mb(inval);
    string out{tmp};
    std::free(tmp);
    return out;
}
const std::wstring pchar2wstring(const char* inval) {
    wchar_t* tmp = mb2wstr(inval);
    wstring out{tmp};
    std::free(tmp);
    return out;
}

const wstring string2wstring(const string& value) {
    return pchar2wstring(value.c_str());
}
const string wstring2string(const wstring& value) {
    return pwchar2string(value.c_str());
}
const wchar_t* char2wchar(const char* value) {
    return pchar2wstring(value).c_str();
}
const char* wchar2char(const wchar_t* value) {
    return pwchar2string(value).c_str();
}