cout << setw与åäö没有正确对齐

Ges*_*tur 10 c++ character-encoding iomanip c++11

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

#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;
}
Run Code Online (Sandbox Code Playgroud)

这会产生以下输出:

COL_A  COL_B  COL_C
ABC    ÅÄÖ ABC
Run Code Online (Sandbox Code Playgroud)

如果我将代码中的"ÅÄÖ"更改为例如"ABC",那么它的工作原理如下:

COL_A  COL_B  COL_C
ABC    ABC    ABC  
Run Code Online (Sandbox Code Playgroud)

为什么会这样?

小智 7

除了使用std::wcout适当的语言环境之外,您可能还必须切换到宽字符串.例如:

void p(std::wstring s, int w)
{
   std::wcout << std::left << std::setw(w) << s;
}

int main(int argc, char const *argv[])
{
   setlocale(LC_ALL, "en_US.utf8");
   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;
}
Run Code Online (Sandbox Code Playgroud)

演示


Dan*_*rey 6

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