Алгоритмизации
Скачать 1.15 Mb.
|
Возможности препроцессораПрепроцессор, как мы уже знаем, это программа предварительной обработки исходного текста программы перед этапом компиляции. Чаще всего препроцессор автоматически вызывается на этапе компиляции, если в исходном тексте обнаружена хотя бы одна его директива. Признаком директивы препроцессора является символ #. При необходимости продолжения директивы в следующей строке текущую строку должен завершать символ '\'. Возможности препроцессора языка Cи: лексемное замещение идентификаторов; макрозамещение; включение файлов исходного текста; условная компиляция; изменение нумерации строк и текущего имени файла. ДирективылексемногозамещенияидентификаторовДиректива определения значения идентификатора (ID): #define IDстрока В результате каждое вхождение в исходный текст элемента ID заменяется на значение элемента строка: #define L_bufs 2048 #define binary int #define WAIT fflush(stdin); getch() #define BEEP sound(800);\ delay(100);\ nosound() Лексемное замещение весьма удобно для сокращения записи повторяющихся фрагментов теста и определения символических констант: #define YES 1 #define NO 2 #define ESC 27 #define Enter 30 которые могут быть в дальнейшем использованы: if (x==ESC) break; BEEP; return(YES); Директиваотмены#undef ID Далее по исходному тексту можно назначить новое значение такого идентификатора. Приложение3 МакрозамещениеМакрозамещение – обобщение лексемного замещения посредством параметризации строки директивы define в виде: #define ID(параметр1,... ) строка между элементом IDи открывающей скобкой пробелы не допускаются. Такой вариант директивы define иногда называют макроопределением. Элемент строка обычно содержит параметры, которые препроцессором будут заменены на фактические аргументы так называемой макрокоманды, записываемой в формате ID(аргумент1,... ) Пример макроопределения и макрокоманд: #define P(X) printf("\n%s",X) . . . char *x; P(x); // Использование макроопределения P(X) P(" НАЧАЛО ОПТИМИЗАЦИИ"); printf("\n%s",x); // Эквивалентные операторы printf("\n%s"," НАЧАЛО ОПТИМИЗАЦИИ"); В строке макроопределений идентификаторы параметров сложных выражений рекомендуется заключать в круглые скобки: #define МАХ(A,B) ((A)>(B)? (A):(B)) #define ABS(X) ((X)<0? –(X):(X)) Потребность в круглых скобках возникает при опасности искажения смысла вложенных выражений из-за действия правил приоритета операций. Пример искажения смысла операций: #define BP(X) X*X . . . int x,y,z; x=BP(y+z); ↔ x=y+z*y+z; ↔ x=y+(z*y)+z; Очевидно, что ошибки будут и при следующих вариантах: #define BP(X) (X*X) #define BP(X) (X)*(X) Безопасный вариант: #define BP(X) ((X)*(X)) Иногда источником ошибок может быть символ «точка с запятой» в конце строки макроопределения: #define BP(X) ((X)*(X)); . . . Приложение3 int x,y,z; x=BP(z)–BP(y); ↔ y=((z)*(z)); –((y)*(y)); Макроопределение отменяется директивой undef. Идентификаторы макроопределений обычно составляют из прописных букв латинского алфавита. Это позволяет отличать макрокоманды от вызова функций. Макрокоманда внешне синтаксически эквивалентна операции вызова функции, но смысл их различен. Функция в программе имеется в одном экземпляре, но на ее вызов тратится время для подготовки параметров и передачи управления. Каждая макрокоманда замещается соответствующей частью макроопределения, но потерь на передачу управления нет. ПодключениефайловисходноготекстаНапомним, что имеются два варианта запроса включения в текущий файл содержимого другого файла. Директива #include < ID_файла> вводит содержимое файла из стандартного каталога (обычно – include), а директива #include " ID_файла" организует последовательный поиск в текущем, системном и стандартном каталогах. Например: #include Рекомендуется описания системных объектов включать из стандартных каталогов и размещать их в начале файла исходного текста программы. Системные объекты в результате получают атрибут области действия «глобальный», что устраняет неоднозначность их описания. УсловнаякомпиляцияДирективы условной компиляции и реализуемые правила включения исходного текста: а) условное включение (аналог работы оператора if): #if<предикат_условия> ТЕКСТ_1 #endif б) альтернативное включение (аналогif-else): #if<предикат_условия> ТЕКСТ_1 #else ТЕКСТ_2 #endif Приложение3 Видыпредикатовусловий:константное_выражение → истина, если его значение ≠0; defID → истина, если IDбыл определен ранее оператором #define; ndefID → истина, если IDне был определен оператором #define. Константное_выражение отделяется от ключевого слова if разделителем, а defи ndef– нет. Пример: #ifdef DEBUG print_state(); #endif Элементы исходного текста «ТЕКСТ_1» или «ТЕКСТ_2» могут содержать любые директивы препроцессора. Примеры: #ifndef EOF #define EOF –1 #endif #if UNIT==CON #include "conproc.c" #else #include "outproc.c" #endif ИзменениенумерациистрокиидентификаторафайлаПо умолчанию диагностические сообщения компилятора привязываются к номеру строки и ID файла исходного текста. Директива #lineномер_строкиID_файла позволяет с целью более приметной привязки к фрагментам текста изменить номер текущей строки и ID файла на новые значения («ID_файла» можно опустить). Приложение4 |