以下代码重现了我的问题:
#include <iostream>
#include <iomanip>
#include <string>
void p(std::string s, int w)
{
std::cout << std::left << std::setw(w) << s;
}
int main(int argc, char const *argv[])
{
p("COL_A", 7);
p("COL_B", 7);
p("COL_C", 5);
std::cout << std::endl;
p("ABC", 7);
p("ÅÄÖ", 7);
p("ABC", 5);
std::cout << std::endl;
return 0;
}
这会产生以下输出:
COL_A COL_B COL_C
ABC ÅÄÖ ABC
如果我将代码中的“ÅÄÖ”改为例如“ABC”,然后它起作用:
COL_A COL_B COL_C
ABC ABC ABC
为什么会这样?
随着imbuing std::cout
使用适当的语言环境,您可能还必须切换到宽字符串。例如:
void p(std::wstring s, int w)
{
std::wcout << std::left << std::setw(w) << s;
}
int main(int argc, char const *argv[])
{
std::locale loc("en_US.UTF-8");
std::wcout.imbue(loc);
p(L"COL_A", 7);
p(L"COL_B", 7);
p(L"COL_C", 5);
std::wcout << std::endl;
p(L"ABC", 7);
p(L"ÅÄÖ", 7);
p(L"ABC", 5);
std::wcout << std::endl;
return 0;
}
演示
之所以会发生这种情况,是因为这些字符(Ä,Ö,...)是可能以UTF-8编码的unicode字符。这意味着每个字符占用几个字节(在您的情况下为两个字节,在一般情况下最多为四个字节)。 setw
OTOH不知道UTF-8 - 它只是计数并因此对齐字节。
问题是你的源代码肯定存储在UTF8中,这意味着每个字母都有2个字节,并且没有相应地设置cout的语言环境。
因此你的cout认为它输出3x2 = 6个字符,并且只增加一个空格以达到预期的7个字符。使用imbue()更改语言环境以将其设置为UTF8。