我无法理解std::string和std::wstring之间的区别。我知道wstring支持Unicode字符等宽字符。我有以下问题:

什么时候应该使用std::wstring而不是std::string?std::string可以保存整个ASCII字符集,包括特殊字符吗?所有流行的C++编译器都支持std::wstring吗?什么是“宽字符”?


当前回答

所以,现在在座的每一位读者都应该清楚地了解事实和情况。如果没有,那么你必须阅读帕塞巴尔非常全面的回答[顺便说一句:谢谢!]。

我的务实结论非常简单:所有C++(和STL)“字符编码”的东西基本上都是残缺不全的。不管是否归咎于微软,这都无济于事。

经过深入调查后,我的解决方案是:

接受,你必须自己负责编码和转换的事情(你会发现很多事情都很琐碎)对任何UTF-8编码字符串使用std::string(仅为typedef std::字符串UTF8String)接受这样一个UTF8String对象只是一个愚蠢但廉价的容器。永远不要直接访问和/或操作其中的字符(不要搜索、替换等)。你可以,但你真的不想浪费时间为多字节字符串编写文本操作算法!即使其他人已经做了如此愚蠢的事情,也不要这样做!顺其自然!(好吧,在某些情况下,这是合理的……只需使用ICU图书馆即可)。对UCS-2编码字符串使用std::wstring(typedef std::wstring UCS2String)-这是一种妥协,也是对WIN32 API引入的混乱的让步)。UCS-2对我们大多数人来说已经足够了(稍后将详细介绍…)。每当需要逐字符访问(读取、操作等)时,请使用UCS2String实例。任何基于字符的处理都应该在非多字节表示中完成。它简单、快速、容易。添加两个实用函数以在UTF-8和UCS-2之间来回转换:UCS2字符串转换为UCS2(const UTF8String&str);UTF8字符串转换为UTF8(常量UCS2String&str);

转换很简单,谷歌应该在这里提供帮助。。。

就是这样。在内存非常宝贵的地方以及所有UTF-8 I/O都使用UTF8String。在必须解析和/或操作字符串的地方使用UCS2String。您可以随时在这两种表示之间进行转换。

替代方案和改进

从&到单字节字符编码(例如ISO-8859-1)的转换可以借助于普通转换表来实现,例如const wchar_ttt_iso88951[256]={0,1,2,…};以及用于转换到UCS2和从UCS2转换的适当代码。如果UCS-2不够,则切换到UCS-4(typedef std::basic_string<uint32_t>UCS2String)

ICU或其他unicode库?

先进的东西。

其他回答

1) 正如Greg所提到的,wstring有助于国际化,这意味着您将以英语以外的语言发布产品

4) 检查此项以获取宽字符http://en.wikipedia.org/wiki/Wide_character

一串wstring?

std::string是在char上模板化的basicstring,而std::wstring是在wchart上模板化。

字符与wchar_t

char应该包含一个字符,通常是8位字符。wchar_t应该包含一个宽字符,然后,事情变得棘手:在Linux上,wchar_t是4字节,而在Windows上,它是2字节。

那么Unicode呢?

问题是char和wchar_t都没有直接绑定到unicode。

在Linux上?

让我们以Linux操作系统为例:我的Ubuntu系统已经支持unicode。当我使用字符串时,它是以UTF-8(即Unicode字符串)本机编码的。以下代码:

#include <cstring>
#include <iostream>

int main()
{
    const char text[] = "olé";


    std::cout << "sizeof(char)    : " << sizeof(char) << "\n";
    std::cout << "text            : " << text << "\n";
    std::cout << "sizeof(text)    : " << sizeof(text) << "\n";
    std::cout << "strlen(text)    : " << strlen(text) << "\n";

    std::cout << "text(ordinals)  :";

    for(size_t i = 0, iMax = strlen(text); i < iMax; ++i)
    {
        unsigned char c = static_cast<unsigned_char>(text[i]);
        std::cout << " " << static_cast<unsigned int>(c);
    }

    std::cout << "\n\n";

    // - - -

    const wchar_t wtext[] = L"olé" ;

    std::cout << "sizeof(wchar_t) : " << sizeof(wchar_t) << "\n";
    //std::cout << "wtext           : " << wtext << "\n"; <- error
    std::cout << "wtext           : UNABLE TO CONVERT NATIVELY." << "\n";
    std::wcout << L"wtext           : " << wtext << "\n";

    std::cout << "sizeof(wtext)   : " << sizeof(wtext) << "\n";
    std::cout << "wcslen(wtext)   : " << wcslen(wtext) << "\n";

    std::cout << "wtext(ordinals) :";

    for(size_t i = 0, iMax = wcslen(wtext); i < iMax; ++i)
    {
        unsigned short wc = static_cast<unsigned short>(wtext[i]);
        std::cout << " " << static_cast<unsigned int>(wc);
    }

    std::cout << "\n\n";
}

输出以下文本:

sizeof(char)    : 1
text            : olé
sizeof(text)    : 5
strlen(text)    : 4
text(ordinals)  : 111 108 195 169

sizeof(wchar_t) : 4
wtext           : UNABLE TO CONVERT NATIVELY.
wtext           : ol�
sizeof(wtext)   : 16
wcslen(wtext)   : 3
wtext(ordinals) : 111 108 233

您将看到char中的“olé”文本实际上由四个字符构成:110、108、195和169(不包括后面的零)。(我将让您学习wchar_t代码作为练习)

因此,在Linux上使用字符时,您通常会在不知道的情况下使用Unicode。由于std::string可以使用字符,所以std::字符串已经可以使用Unicode。

请注意,std::string与C字符串API一样,会认为“olé”字符串有4个字符,而不是3个字符。因此,在截断/播放unicode字符时应谨慎,因为UTF-8中禁止某些字符组合。

在Windows上?

在Windows上,这有点不同。在Unicode出现之前,Win32必须支持许多使用字符和世界各地产生的不同字符集/代码页的应用程序。

因此,他们的解决方案是一个有趣的解决方案:如果应用程序使用字符,那么字符字符串将使用机器上的本地字符集/代码页编码/打印/显示在GUI标签上,这在很长一段时间内都不可能是UTF-8。例如,在法语本地化的Windows中,“olé”将是“olé”,但在cyrillic本地化的Windows上则会有所不同(如果使用Windows-1251,则为“olй”)。因此,“历史应用程序”通常仍将以旧的方式工作。

对于基于Unicode的应用程序,Windows使用wchar_t,其宽度为2字节,并以UTF-16编码,UTF-16以2字节字符为Unicode编码(或者至少是UCS-2,它只是缺少代理对,因此缺少BMP之外的字符(>=64K))。

使用字符的应用程序称为“多字节”(因为每个字形由一个或多个字符组成),而使用wchar_t的应用程序则称为“宽字符”(因为每一个字形由一或两个wchar_t)。有关详细信息,请参阅MultiByteToWideChar和WideCharToMultiByteWin32转换API。

因此,如果你在Windows上工作,你很想使用wchar_t(除非你使用一个隐藏它的框架,如GTK或QT…)。事实是,在幕后,Windows使用wchar_t字符串,所以即使是历史应用程序,在使用SetWindowText()(用于在Win32 GUI上设置标签的低级API函数)等API时,也会将其字符字符串转换为wchar_t。

内存问题?

UTF-32是每个字符4个字节,因此没有什么可添加的,只要UTF-8文本和UTF-16文本总是比UTF-32文本使用更少或相同的内存量(通常更少)。

如果存在内存问题,那么您应该知道,与大多数西方语言相比,UTF-8文本使用的内存将少于相同的UTF-16文本。

尽管如此,对于其他语言(中文、日语等),UTF-8使用的内存将与UTF-16相同,或者略大。

总而言之,UTF-16通常每个字符使用2个字节,有时使用4个字节(除非您正在处理某种深奥的语言字形(克林贡语?精灵语?),而UTF-8将使用1到4个字节。

看见https://en.wikipedia.org/wiki/UTF-8#Compared_to_UTF-16获取更多信息。

结论

什么时候应该在std::string上使用std::wstring?在Linux上?几乎从未(§)。在Windows上?几乎总是(§)。跨平台代码?取决于您的工具包。。。(§):除非您使用的工具包/框架另有说明std::string可以保存所有ASCII字符集,包括特殊字符吗?注意:std::string适合保存“binary”缓冲区,而std::wstring不是!在Linux上?对在Windows上?只有特殊字符可用于Windows用户的当前区域设置。编辑(Johann Gerell发表评论后):一个std::字符串将足以处理所有基于字符的字符串(每个字符都是从0到255的数字)。但是:ASCII应该从0到127。较高的字符不是ASCII码。从0到127的字符将被正确保存从128到255的字符将根据您的编码(unicode、非unicode等)而有意义,但只要以UTF-8编码,它将能够保存所有unicode字形。几乎所有流行的C++编译器都支持std::wstring吗?大多数情况下,除了移植到Windows的基于GCC的编译器。它适用于我的g++4.3.2(在Linux下),我从Visual C++6开始在Win32上使用Unicode API。宽字符到底是什么?在C/C++上,它是一种wchar_t编写的字符类型,比简单的char字符类型更大。它应该用于放置索引(如Unicode字形)大于255(或127,取决于…)的字符。

如果应用程序不满足256个不同的字符,则可以选择使用宽字符(超过8位)或可变长度编码(C++术语中的多字节编码),如UTF-8。宽字符通常比可变长度编码需要更多的空间,但处理速度更快。处理大量文本的多语言应用程序通常在处理文本时使用宽字符,但在将文本存储到磁盘时将其转换为UTF-8。

字符串和wstring之间的唯一区别是它们存储的字符的数据类型。字符串存储的字符大小保证至少为8位,因此您可以使用字符串来处理例如ASCII、ISO-8859-15或UTF-8文本。该标准没有说明字符集或编码。

实际上,每个编译器都使用一个字符集,其前128个字符与ASCII对应。使用UTF-8编码的编译器也是如此。在UTF-8或其他可变长度编码中使用字符串时,需要注意的重要一点是,索引和长度是以字节而不是字符来度量的。

wstring的数据类型是wchar_t,其大小在标准中没有定义,除了它必须至少与一个字符一样大,通常是16位或32位。wstring可用于处理实现定义的宽字符编码中的文本。因为标准中没有定义编码,所以在字符串和wstring之间进行转换并不简单。也不能假设wstring具有固定长度编码。

如果您不需要多语言支持,那么可以只使用常规字符串。另一方面,如果您正在编写图形应用程序,则API通常只支持宽字符。然后,您可能希望在处理文本时使用相同的宽字符。请记住,UTF-16是一种可变长度编码,这意味着您不能假定length()返回字符数。如果API使用固定长度编码,例如UCS-2,则处理变得容易。在宽字符和UTF-8之间进行转换很难以可移植的方式进行,但话说回来,您的用户界面API可能支持这种转换。

如果保持字符串的可移植性,则可以使用tstring,tchar。这是很久以前广泛使用的技术。在这个示例中,我使用了自定义的TCHAR,但您可以在internet上找到linux的TCHAR.h实现。

这个想法意味着windows上使用wstring/wchar_t/UTF-16,Linux上使用string/char/UTF-8(或ASCII..)。

在下面的示例中,英语/日语多字节混合字符串的搜索在两个windows/linux平台上都运行良好。

#include <locale.h>
#include <stdio.h>
#include <algorithm>
#include <string>
using namespace std;

#ifdef _WIN32
    #include <tchar.h>
#else
    #define _TCHAR char
    #define _T 
    #define _tprintf printf
#endif

#define tstring basic_string<_TCHAR>

int main() {
    setlocale(LC_ALL, "");
    tstring s = _T("abcあいうえおxyz");

    auto pos = s.find(_T("え"));
    auto r = s.substr(pos);
    _tprintf(_T("r=%s\n"), r.c_str());
}

当您希望使用Unicode字符串而不仅仅是ascii时,这有助于国际化是的,但它与0的配合不好没有意识到宽字符是编译器特有的处理unicode字符固定长度表示的方式,对于MSVC,它是2字节字符,对于gcc,我理解它是4字节。以及+1表示http://www.joelonsoftware.com/articles/Unicode.html