Урок 2. Очистка экрана, проблема с кириллицей и др.

Не всё, что изначально кажется простым,
окажется таковым в конце.

 

В прошлый раз мы написали программу, которая здоровается с вами, выводя на экран приветственное сообщение «Привет!». Однако переключаясь в окно, где выводятся результаты выполнения программы, мы видели не только сообщение «Привет!», но и целый ряд служебных сообщений, которые Free Pascal выводит во время своего запуска.

Хотелось бы, чтобы наша программа перед выводом каких-либо сообщений очищала экран. Как это сделать? Очень просто: нужно добавить в программу вызов команды (процедуры) ClrScr (сокращение от английского «clear screen» — «очистить экран»).

Если ваша предыдущая программа hello.pas закрыта, то откройте её, нажав клавишу F3 (File | Open). Напоминаю, что все наши программы мы сохраняем в папке «C:\FPC\3.0.0\work».

Чтобы не затереть файл hello.pas нашими модификациями, сохраним открытый файл hello.pas под другим именем. Откройте меню «File» (Alt + F) и нажмите «Save as...». В поле «Name» напечатайте helloClr.pas (Clr — сокр. от «clear» — очищать) и нажмите «OK», чтобы сохранить программу под новым именем. После этого модифицируйте текст программы так, чтобы он соответствовал нижеприведенному варианту:

 


program helloClr;

begin
   ClrScr;

   Writeln( 'Привет!' );
end.

 

Запустите программу, нажав Ctrl + F9.

Упс! Программа не запускается. Что говорит компилятор? (Напомню, что компилятор — Free Pascal — переводит нашу программу с языка Паскаль в машинные инструкции, понятные компьютеру). Компилятор сообщает, что у нас 3 ошибки. Одна с пометкой Error и две с пометкой Fatal. Но мы, глядя на текст сообщений, уже видим, что ошибка на самом деле ровно одна — Error: Identifier not found "ClrScr" — Ошибка: идентификатор "ClrScr" не найден.

Идентификатор — это имя. Компилятор не понимает, что такое ClrScr. Вспомним, в прошлый раз подобная ошибка у нас была, когда мы пропустили букву « в слове Writeln. Компилятор не знал, что такое Writln и из-за этого ругался. Может быть, мы допустили ошибку в названии процедуры (команды) ClrScr? Смотрим внимательно. Нет ошибок? И тем не менее компилятор не понимает, что такое ClrScr.

Да, так и есть. Компилятор не знает, что такое ClrScr

Но почему тогда он знает Writeln и не знает ClrScr? Дело в том, что процедура (команда) Writeln относится к базовым процедурам Паскаля, а вот алгоритм работы процедуры ClrScr описан отдельно. Отдельно — означает в отдельном модуле (файле). И для того, чтобы компилятор Free Pascal знал, как работает эта процедура, вы должны указать в своей программе, что используете модуль Crt. Так называется модуль, в котором описан алгоритм процедуры ClrScr. Или как еще говорят: реализована процедура ClrScr.

Добавим к нашей программе недостающую строку «uses Crt;»:


program helloClr;

uses
   Crt;

begin
   ClrScr;

   Writeln( 'Привет!' );
end.

 

Uses — по-английски означает «использует». Модуль Crt — тот самый модуль, в котором реализована процедура ClrScr.

Теперь мы можем запустить нашу программу (Ctrl + F9):

Что ж, на этот раз наша программа запустилась, и даже очистился экран, вот только вместо «Привет!» мы видим на экране какую-то белиберду. Т. е. Free Pascal не отображает русские буквы (кириллицу), если подключен модуль Crt и используется процедура ClrScr (эксперимент показывает, что даже если удалить строку с вызовом процедуры ClrScr, проблема с отображением кириллицы все равно остается).

Что делать?

Открываем браузер, и в поисковой строке гугла вбиваем «free pascal clrscr не отображает русские буквы»:

 

Первая же ссылка приводит нас на форум, где мы находим решение:

 

Чтобы исправить проблему с отображением русских букв (кириллицы), сохраняем нашу программу (F2 — File | Save), закрываем Free Pascal (Alt + X — File | Exit), и щелкаем правой кнопкой мыши на ярлыке запуска Free Pascal, расположенном на нашем Рабочем столе.

Открываем пункт «Свойства» и переходим на закладку «Шрифт».

Возвращаем «Точечные шрифты», размер выставляем «10 x 18». Нажимаем «ОК».

И снова запускаем Free Pascal:

Как видите, вид окна поменялся.

Запускаем нашу программу (Ctrl + F9), нажимаем Alt + F5, чтобы переключиться в окно результатов работы программы, и видим результат:

Экран очищен, на экран выведено сообщение «Привет!».

 

Краткое содержание урока

✔  Мы узнали как правильно закрывать Free Pascal (раньше вы, скорее всего, просто нажимали на крестик в правом верхнем углу окна):

  • Alt + X (File | Exit) — закрыть Free Pascal.

✔  Дополнили нашу программу «Привет!» очисткой экрана:

program helloClr;

uses
   Crt;

begin
   ClrScr;

   Writeln( 'Привет!' );
end.

✔  Узнали новое ключевое слово языка Паскаль:

  • uses — ключевое слово, после которого перечисляются все модули, используемые нами в программе.

✔  Познакомились с еще одной процедурой:

  • ClrScr — очищает экран от текста.

✔  Узнали о существовании модуля Crt.

✔  И наконец, мы впервые столкнулись с проблемой на уровне инструмента (Free Pascal): когда после подключения модуля Crt у нас перестали отображаться русские буквы (кириллица) в окне результатов. Мы не разобрались, почему это происходит, т. к. причины скрыты внутри модуля Crt. Можно ли выяснить причины, по которым у нас возникла проблема с кириллицей? Чисто теоретически, да. Но для этого нам бы понадобились знания, которые выходят за рамки наших уроков. Поэтому мы воспользовались обходным решением (workaround) — поменяли используемый в программе шрифт Lucida Console на точечный шрифт.

 

И вот здесь я хочу остановиться подробнее

При реализации нашей программы у нас возникла проблема с кириллицей, которую мы решили. Казалось бы — вопрос снят: экран очищается, сообщение отображается. Но обратите внимание, что мы отказались от использования шрифта Lucida Console, который смотрится гораздо опрятнее точечного шрифта. Т. е. наше решение (или правильнее, обходное решение) далеко от идеального.

Приведу аналогию. Представьте себе, что вы — работник дорожной службы, которому поручили на такой-то улице нарисовать «зебру» и поставить соответствующие знаки пешеходного перехода. Вы приехали на заданную улицу и обнаружили, что в том месте, где вам надо рисовать «зебру», из-под асфальта бьет фонтан воды, и образовалась лужа. Какие у вас есть варианты действий:

  1. Выяснить откуда вода (скорее всего, прорвало трубу), устранить течь, высушить лужу и уже после этого рисовать «зебру».
  2. Нарисовать «зебру» 50 метрами ниже (или выше) — там, где нет воды, и где дорожные условия позволяют это сделать.

Так вот, второй вариант — это и есть обходное решение (workaround). Оно может иметь серьезные недостатки. Например, школа находится именно там, где и надо было изначально рисовать «зебру», и если вы нарисуете её ниже (выше), то далеко не все школьники дойдут до этой зебры, а значит, на изначальном участке дороги будет повышенная аварийность из-за школьников, перебегающих дорогу в неположенном месте.

А что нужно, чтобы пойти по пути первого варианта? Другие компетенции. Если вы не владеете нужными навыками (и инструментами) — у вас нет экскаватора, чтобы добраться до трубопровода, сварочного аппарата и т. д. — то вам нужно ждать, когда другие люди решат возникшую проблему, прежде чем вы сможете нарисовать «зебру» именно там, где вам надо. В нашем примере это означает, что вы должны написать письмо разработчикам Free Pascal, описать возникшую проблему (некорректное отображение кириллицы после подключения модуля Crt), и ждать, когда разработчики устранят эту ошибку в модуле Crt. На всё это может уйти неопределенное время. Или разработчикам Free Pascal будет вообще не до этой проблемы, и ваша «лужа» так никуда и не денется.

Другой сценарий: обладая нужными навыками, устранить самому эту проблему в модуле Crt. А для этого нужно знать, что такое дизассемблирование, обратная разработка (reverse engineering) и т. д.

Идеальный вариант — это, конечно, когда лужу устраняют те, кто за нее отвечают. Т. е. разработчики Free Pascal. Но мы живем не в идеальном мире, и поэтому нам иногда приходится использовать обходные решения.

И последнее: когда ваши программы перестанут вмещаться на одну страницу экрана, и будут занимать сотни, а то и больше, страниц, у вас могут тоже возникать подобные ситуации. Когда что-то идет не так и нужно найти причины проблемы, а потом её решение. И вот тут может возникнуть соблазн «передвинуть «зебру» на 50 метров выше или ниже» (использовать обходное решение) вместо того, чтобы разбираться, почему возникла проблема и как её устранить. В случае, когда речь идет о вашей зоне ответственности, такой подход, в целом, недопустим (!). Вы должны всегда понимать, что и почему происходит в вашей программе. Если оказывается, что программа ведет себя не так, как вы это задумывали, надо разбираться, почему она себя так ведет (откуда лужа?), а не цепляться за первое попавшееся решение, которое вроде бы «решает» вашу проблему.

 

Задания

  1. Что будет выведено на экране после запуска следующей программы?
program helloClr;

uses
   Crt;

begin
   ClrScr;
   ClrScr;

   Writeln( 'Привет!' );
end.

 

  1. А после запуска такой программы?
program helloClr;

uses
   Crt;

begin
   ClrScr;
   Writeln( 'Привет!' );

   ClrScr;
   Writeln( 'Пока!' );
end.

 

  1. Исправьте все ошибки в следующей программе, и запустите исправленный вариант:
progrann helloClr;

use
   Crt;

begin
   ClrScr
   Writeln( Привет!' );
end;

Языковая подсказка:

«illegal character» переводится как «некорректный символ».

 

Задания повышенной сложности

  1. Напишите программу, которая выведет на чистый экран текст программы hello.pas (см. урок 1).

Подсказки:

1) чтобы вывести на экран пустую строку, используйте команду «Writeln;» (без скобок и одинарных кавычек);

2) для отступов используйте по 3 пробела;

3) чтобы вывести одинарные кавычки на экран, используйте по 2 одинарных кавычки, например: «Writeln( '''Привет''' );» напечатает на экране не просто Привет, а 'Привет'.

Непонятно почему в скобках по 3 кавычки? Смотрите: самая левая и самая правая одинарные кавычки обрамляют тот текст, который мы хотим вывести на экран, например:

Writeln( 'Кафе Огонёк' );

А что делать, если нам нужно в сообщении вывести одинарные кавычки? Ведь они задействованы в качестве служебных символов, по которым Free Pascal определяет, где наша строка в программе начинается, а где заканчивается. Вот поэтому Free Pascal говорит: хочешь вывести в своей строке одну одинарную кавычку, поставь их две (!) друг за дружкой, и я буду знать, что это не конец строки, а всего лишь обычная одинарная кавычка, которую ты хочешь напечатать.

Например, вот так:

Writeln( 'Кафе ''Огонек'' открывается сегодня' );

Команда выше выведет на экран сообщение:

Кафе 'Огонек' открывается сегодня

 

  1. Подумайте, легко ли написать программу, которая выведет на экран саму себя, т. е. свой собственный текст (такая программа называется куайн — по-английски, quine)?