问题 cout


以下代码重现了我的问题:

#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  

为什么会这样?


9975
2018-03-21 22:47


起源

尝试打印字符串的长度。 - Hurkyl


答案:


随着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;
}

演示


6
2018-03-21 22:58





之所以会发生这种情况,是因为这些字符(Ä,Ö,...)是可能以UTF-8编码的unicode字符。这意味着每个字符占用几个字节(在您的情况下为两个字节,在一般情况下最多为四个字节)。 setw OTOH不知道UTF-8 - 它只是计数并因此对齐字节。


5
2018-03-21 22:55





问题是你的源代码肯定存储在UTF8中,这意味着每个字母都有2个字节,并且没有相应地设置cout的语言环境。

因此你的cout认为它输出3x2 = 6个字符,并且只增加一个空格以达到预期的7个字符。使用imbue()更改语言环境以将其设置为UTF8。


2
2018-03-21 22:55