Wygląda na to, że jest to błąd w Perlu. Myślałem, że to błąd na stronie kodowej Windows 65001, który nie jest obsługiwany przez konsolę, ale w końcu zrobiłem programy testowe w C i Perlu, a problem nie występuje w wersji C. Dzieje się tak bez względu na to, gdzie w linii występuje znak Unicode, ale linia, którą drukujesz, musi być szersza niż obsługuje konsola.
Oto mój program w C:
#include "stdafx.h"
#include "Windows.h"
int _tmain(int argc, _TCHAR* argv[])
{
BOOL b = SetConsoleOutputCP(65001);
printf("set console output codepage returned %d\n", b);
printf("cαfe\n");
printf("1234567890 café\n");
printf("1234567890 1234567890 cαfe\n");
printf("1234567890 1234567890 1234567890 café\n");
printf("1234567890 1234567890 1234567890 1234567890 cαfe\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 café\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n");
printf("1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n");
return 0;
}
I tu jest mój program Perl:
#
use utf8;
binmode STDOUT, ':utf8';
printf STDOUT "cαfe\n";
printf STDOUT "1234567890 café\n";
printf STDOUT "1234567890 1234567890 cαfe\n";
printf STDOUT "1234567890 1234567890 1234567890 café\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 cαfe\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 café\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 café\n";
printf STDOUT "1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 1234567890 cαfe\n";
UPDATE
Nie myliłem się, z pomocą niektórych faceci na #perl na irc.perl.org to okazuje się być błędem w API Microsoftu. WriteFile
jest udokumentowany, aby zwrócić liczbę zapisanych bajtów, ale zwraca liczbę znaków zapisanych, która zależy od strony kodowej. A bug was filed in March 2010.
Istnieje więcej dyskusji in the MSDN forums.
UPDATE 2
napisałem blogu Michael Kaplan, w "Sorting it all out", o tym problemie, a on odpowiedział z artykułu zatytułowanego "Hidden in plain site: a purloined letter kind of a bug report". Jest ekspertem od internacjonalizacji firmy Microsoft, więc na pewno znajdziesz tam pewne spostrzeżenia ...
Brak pomysłu; nie dzieje się dla mnie. Czy możesz nam powiedzieć coś o środowisku, w którym to uruchamiasz? – ysth