13

次のコードは私の問題を再現します:

#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  

なぜこうなった?

4

5 に答える 5

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

デモ

于 2015-03-21T22:58:35.913 に答える
6

これらの文字 (Ä、Ö、...) は、UTF-8 でエンコードされている可能性が高い Unicode 文字であるために発生します。これは、各文字が数バイト (あなたの場合は 2、一般的な場合は最大 4) を占めることを意味します。setwOTOH は UTF-8 について認識していません。つまり、バイトをカウントしてアライメントするだけです。

于 2015-03-21T22:55:04.923 に答える
2

問題は、ソース コードが確実に UTF8 (ÅÄÖ の文字ごとに 2 バイトを意味する) で保存されており、cout のロケールがそれに応じて設定されていないことです。

したがって、cout は 3x2=6 文字を出力すると考え、予想される 7 に到達するためにスペースを 1 つだけ追加します。imbue() を使用してロケールを変更し、UTF8 に設定します。

于 2015-03-21T22:55:00.410 に答える