Обработка двоичных файлов
Проблема
Операционная система отличает текстовые файлы от двоичных. Как это сделать в программе?
Решение
Вызовите функцию
binmode для файлового манипулятора:
binmode(МАНИПУЛЯТОР);
Комментарий
Не существует единого мнения по поводу того, что является строкой текстового файла; текстовые
символы одного компьютера могут превратиться в двоичную белиберду на другом. Но даже если все
станут пользоваться кодировкой ASCII вместо EBCDIC, Rad50 или Unicode, могут возникнуть затруднения.
Как говорилось во введении, конкретного символа перевода строки не существует. Это чисто
абстрактное понятие, которое поддерживается операционной системой, стандартными библиотеками,
драйверами устройств и Perl.
В Unix или Plan9 "\n" представляет физическую последовательность "\cJ" (служебная
последовательность Perl, соответствующая Ctrl+J). Однако на терминале, не работающем в «чистом»
(raw) режиме, нажатие на клавишу Enter генерирует код "\сМ" (возврат курсора),
транслируемый в "\cJ", а выходной код "\cJ" транслируется в "\cM\cJ". Подобные странности
характерны не для обычных файлов, а лишь для терминальных устройств, и обрабатываются строго
на уровне драйвера устройства.
На Мас код "\n" обычно представляется "\сМ"; чтобы жизнь была интереснее (а также из-за
стандартов, требующих различий между "\n" и "\r"), "\r" соответствует "\cJ". Такая интерпретация
в точности противоположна стандартам UNIX, Plan9, VMS, CP/M... словом, почти всем.
Следовательно, программисты Мас, которые пишут файлы для других систем или общаются с ними
по сети, должны проявлять осторожность. Если отправить "\n", вы получите "\сМ", a "\cJ" исчезнет.
Многие сетевые службы предпочитают отправлять и принимать в качестве разделителя строк
последовательность "\cM\cJ", однако большинство позволяет ограничиться простым "\cJ".
В VMS, DOS и их производных "\n" также представляет "\cJ", по аналогии с Unix и Plan9.
С терминальной точки зрения UNIX и DOS ведут себя одинаково: при нажатии пользователем клавиши
Enter генерируется "\сМ", однако в программу поступает уже "\n", то есть "\cJ". Код "\n",
переданный терминалу, превращается в "\cM\cJ".
Эти странные преобразования выполняются и с файлами Windows. В текстовых файлах DOS
каждая строка завершается двумя символами, "\cM\cJ". Последний блок файла содержит код "\cZ",
определяющий окончание текста. В таких системах при записи строки "bad news\n" файл будет
содержать "bad news\cM\cJ", как при выводе на терминал.
Но при чтении строк в таких системах происходят еще более странные вещи. Файл содержит
"bad news\cM\cJ" — строку, состоящую из 10 байт. При чтении ваша программа не получит ничего,
кроме "bad news\n", где "\n" — виртуальный символ перевода строки, то есть "\cJ". Следовательно,
от него можно избавиться одним вызовом
chop или
chomp. Однако при этом приходится обманывать
бедную программу и внушать ей, что из файла было прочитано всего 9 байт. Если прочитать 10
таких строк, она будет полагать, что из файла было прочитано 90 байт, хотя в действительности
смещение будет равно 100. Из-за этого для определения текущей позиции всегда следует использовать
функцию tell. Простой подсчет прочитанных байтов не подходит.
Такое наследие старой файловой системы СР/М, в которой хранились лишь сведения о количестве
блоков, но не о размере файлов, бесит программистов уже несколько десятилетий, и конца-края
этому не видно. Ведь DOS была совместима с файловым форматом СР/М, Windows — с форматом DOS,
a NT — с форматом Windows. Грехи отцов преследуют потомков в четвертом поколении.
Впрочем, проблему одиночного "\n" можно обойти — достаточно сообщить Perl (и операционной
системе), что вы работаете с двоичными данными. Функция binmode означает, что прочитанные
или записанные через конкретный манипулятор данные не должны преобразовываться по правилам,
установленным в системе для текстовых файлов.
$gifname = "picture.gif";
open(GIF, $gifname) or die "can't open $gifname: $!";
binmode(GIF); # Теперь DOS не преобразует двоичные
# входные данные GIF
binmode(STDOUT); # Теперь DOS не преобразует двоичные
# выходные данные STDOUT
while (read(GIF, $buff, 8*2**10)) {
print STDOUT $buff;
}
Вызов binmode в системах, где отличия между текстовыми и двоичными файлами несущественны
(в том числе UNIX, Mac и Plan9), не принесет никакого вреда. Однако несвоевременный вызов
функции в других системах (включая MVS, VMS и всех разновидностей DOS) может исказить содержимое файлов.
Если функция binmode не используется, в данных, прочитанных с помощью <>, строковый
терминатор системы заменяется на "\n", даже если $/ было присвоено другое значение.
Аналогично, любой "\n", выводимый через манипулятор функцией
print, превращается в строковый
терминатор данной системы. Дополнительные сведения приведены во введении.
Если вы хотите, чтобы прочитанные данные совпадали с содержимым файла байт в байт, и
при этом работаете в одной из перечисленных странных систем, — вызовите binmode. Конечно,
если вы захотите использовать их, вам придется присвоить $/ настоящий разделитель записей.
См. такжеОписание функции open и
binmode
Proverte kod v komentariyah gde pro list tam oshibki detskie
Оставить комментарий:
|
|