Лекции.ИНФО


Если имя файла заключено в угловые скобки, то поиск файла будет осуществляться в одном или нескольких специальных каталогах, определенных конкретной реализацией.



Если же имя файла заключено в кавычки, поиск файла выполняется, как правило, в текущем каталоге (что также определено конкретной реализацией). Во многих случаях это означает поиск текущего рабочего каталога. Если заданный файл не найден, поиск повторяется с использованием первого способа (как если бы имя файла было заключено в угловые скобки). Чтобы ознакомиться с подробностями, связанными с различной обработкой директивы #include в случае использования угловых скобок и двойных кавычек, обратитесь к руководству пользователя, прилагаемому к вашему компилятору. Инструкции #include могут быть вложенными в другие включаемые файлы.

Директивы условной компиляции

Существуют директивы, которые позволяют избирательно компилировать части исходного кода. Этот процесс, именуемый условной компиляцией, широко используется коммерческими фирмами по разработке программного обеспечения, которые создают и поддерживают множество различных версий одной программы.

Директивы #if, #else, #elif и #endif

Директивы #if, #ifdef, #ifndef, #else, #elif и #endif — это директивы условной компиляции.

Главная идея состоит в том, что если выражение, стоящее после директивы #if оказывается истинным, то будет скомпилирован код, расположенный между нею и директивой #endif в противном случае данный код будет опущен. Директива #endif используется для обозначения конца блока #if.

Общая форма записи директивы #if выглядит так.

#if константное_выражение

 последовательность инструкций

#endif

Если константное_выражение является истинным, код, расположенный непосредственно за этой директивой, будет скомпилирован. Рассмотрим пример.

// Простой пример использования директивы #if.

#include <iostream>

using namespace std;

#define MAX 100

Int main()

{

 #if MAX>10

  cout << "Требуется дополнительная память";

 #endif

 // ...

 return 0;

}

При выполнении эта программа отобразит сообщение Требуется дополнительная память на экране, поскольку, как определено в программе, значение константы МАХ больше 10. Этот пример иллюстрирует важный момент: Выражение, которое стоит после директивы #if, вычисляется во время компиляции. Следовательно, оно должно содержать только идентификаторы, которые были предварительно определены, или константы. Использование же переменных здесь исключено.

Поведение директивы #else во многом подобно поведению инструкции else, которая является частью языка C++: она определяет альтернативу на случай невыполнения директивы #if. Чтобы показать, как работает директива #else, воспользуемся предыдущим примером, немного его расширив.

// Пример использования директив #if/#else.

#include <iostream>

using namespace std;

#define MAX 6

Int main()

{

 #if MAX>10

  cout << "Требуется дополнительная память.");

 #else

  cout << "Достаточно имеющейся памяти.";

 #endif

 // . . .

 return 0;

}

В этой программе для имени МАХ определено значение, которое меньше 10, поэтому #if-ветвь кода не скомпилируется, но зато скомпилируется альтернативная #else-ветвь. В результате отобразится сообщение Достаточно имеющейся памяти..

Обратите внимание на то, что директива #else используется для индикации одновременно как конца #if-блока, так и начала #еlse-блока. В этом есть логическая необходимость, поскольку только одна директива #endif может быть связана с директивой #if.

Директива #elif эквивалентна связке инструкций else-if и используется для формирования многозвенной схемы if-else-if, представляющей несколько вариантов компиляции. После директивы #elif должно стоять константное выражение. Если это выражение истинно, следующий блок кода скомпилируется, и никакие другие #elif-выражения не будут тестироваться или компилироваться. В противном случае будет проверено следующее по очереди #elif-выражение. Вот как выглядит общий формат использования директивы #elif.

#if выражение

 последовательность инструкций

#elif выражение 1

 последовательность инструкций

#elif выражение 2

 последовательность инструкций

#еlif выражение 3

 последовательность инструкций

// . . .

#elif выражение N

 последовательность инструкций

#endif

Например, в этом фрагменте кода используется идентификатор COMPILED_BY, который позволяет определить, кем компилируется программа.

#define JOHN 0

#define BOB 1

#define TOM 2

#define COMPILED_BY JOHN

#if COMPILED_BY == JOHN

 char who[] = "John";

#elif COMPILED_BY == BOB

 char who[] = "Bob";

#else

 char who[] = "Tom";

#endif

Директивы #if и #elif могут быть вложенными. В этом случае директива #endif, #else или #elif связывается с ближайшей директивой #if или #elif. Например, следующий фрагмент кода совершенно допустим.

#if COMPILED_BY == BOB

 #if DEBUG == FULL

  int port = 198;

 #elif DEBUG == PARTIAL

  int port = 200;

 #endif

#else

 cout << "Боб должен скомпилировать код" << "для отладки вывода данных.";

#endif

Директивы #ifdef и #ifndef

Директивы #ifdef и #ifndef предлагают еще два варианта условной компиляции, которые можно выразить как "если определено" и "если не определено" соответственно.

Общий формат использования директивы #ifdef таков.

#ifdef макроимя

 последовательность инструкций

#endif

Если макроимя предварительно определено с помощью какой-нибудь директивы #define, то последовательность инструкций, расположенная между директивами #ifdef и #endif, будет скомпилирована.

Общий формат использования директивы #ifndef таков.

#ifndef макроимя

 последовательность инструкций

#endif

Если макроимя не определено с помощью какой-нибудь директивы #define, то последовательность инструкций, расположенная между директивами #ifdef и #endif, будет скомпилирована.

Как директива #ifdef, так и директива #ifndef может иметь директиву #else или #elif. Рассмотрим пример.

#include <iostream>

using namespace std;

#define TOM

Int main()

{

 #ifdef TOM

  cout << "Программист Том.";

 #else

  cout << "Программист неизвестен.";

 #endif

 #ifndef RALPH

  cout << "Имя RALPH не определено.";

 #endif

 return 0;

}

При выполнении эта программа отображает следующее.

Программист Том.

Имя RALPH не определено.

Но если бы идентификатор ТОМ был не определен, то результат выполнения этой программы выглядел бы так.

Программист неизвестен.

Имя RALPH не определено.

И еще. Директивы #ifdef и #ifndef можно вкладывать точно так же, как и директивы #if.

Директива #undef

Директива #undef используется для удаления предыдущего определения некоторого макроимени. Ее общий формат таков.

#undef макроимя

Рассмотрим пример.

#define TIMEOUT 100

#define WAIT 0

 // . . .

#undef TIMEOUT

#undef WAIT

Здесь имена TIMEOUT и WAIT определены до тех пор, пока не выполнится директива #undef.

Основное назначение директивы #undef — разрешить локализацию макроимен для тех частей кода, в которых они нужны.

Использование оператора defined

Помимо директивы #ifdef существует еще один способ выяснить, определено ли в программе некоторое макроимя. Для этого можно использовать директиву #if в сочетании с оператором времени компиляции defined. Например, чтобы узнать, определено ли макроимя MYFILE, можно использовать одну из следующих команд препроцессорной обработки.

#if defined MYFILE

Или

#ifdef MYFILE

При необходимости, чтобы реверсировать условие проверки, можно предварить оператор defined символом "!". Например, следующий фрагмент кода скомпилируется только в том случае, если макроимя DEBUG не определено.

#if !defined DEBUG

 cout << "Окончательная версия!";

#endif

О роли препроцессора

Препроцессор C++ — прямой потомок препроцессора языка С, причем без каких-либо усовершенствований. Однако его роль в C++ намного меньше роли, которую играет препроцессор в С. Дело в том, что многие задачи, выполняемые препроцессором в С, реализованы в C++ в виде элементов языка. Страуструп тем самым выразил свое намерение сделать функции препроцессора ненужными, чтобы в конце концов от него можно было бы совсем освободить язык.

На данном этапе препроцессор уже частично избыточен. Например, два наиболее употребительных свойства директивы #define были заменены инструкциями C++. В частности, ее способность создавать константное значение и определять макроопределение, действующее подобно функциям, сейчас совершенно избыточна. В C++ есть более эффективные средства для выполнения этих задач. Для создания константы достаточно определить const-переменную. А с созданием встраиваемой (подставляемой) функции вполне справляется спецификатор inline. Оба эти средства лучше работают, чем соответствующие механизмы директивы #define.

Приведем еще один пример замены элементов препроцессора элементами языка. Он связан с использованием однострочного комментария. Одна из причин его создания — разрешить "превращение" кода в комментарий. Как вы знаете, комментарий, использующий /*...*/-стиль, не может быть вложенным. Это означает, что фрагменты кода, содержащие /*...*/-комментарии, одним махом "превратить в комментарий" нельзя. Но это можно сделать с //-комментариями, окружив их /*...*/-символами комментария. Возможность "превращения" кода в комментарий делает использование таких директив условной компиляции, как #ifdef, частично избыточным.

Директива #line

Директива #line изменяет содержимое псевдопеременных _ _LINE_ _ и _ _FILE_ _.

Директива #line используется для изменения содержимого псевдопеременных _ _LINE_ _ и _ _FILE_ _, которые являются зарезервированными идентификаторами (макроименами). Псевдопеременная _ _LINE_ _ содержит номер скомпилированной строки, а псевдопеременная _ _FILE_ _— имя компилируемого файла. Базовая форма записи этой команды имеет следующий вид.

#line номер "имя_файла"

Здесь номер — это любое положительное целое число, а имя_файла — любой допустимый идентификатор файла. Значение элемента номер становится номером текущей исходной строки, а значение элемента имя_файла— именем исходного файла. Имя_файла— элемент необязательный. Директива #line используется, главным образом, в целях отладки и в специальных приложениях.

Например, следующая программа обязывает начинать счет строк с числа 200. Инструкция cout отображает номер 202, поскольку это — третья строка в программе после директивной инструкции #line 200.

#include <iostream>

using namespace std;

#line 200 // Устанавливаем счетчик строк равным 200.









Читайте также:

  1. Adjective and adverb. Имя прилагательное и наречие. Степени сравнения.
  2. Cosa Nostra история сицилийской мафии
  3. D. ОХРАНА ГЕОГРАФИЧЕСКИХ УКАЗАНИЙ НА МЕЖДУНАРОДНОМ УРОВНЕ В СИЛУ ПОЛОЖЕНИЙ ДВУСТОРОННИХ СОГЛАШЕНИЙ
  4. F73.04 Умственная отсталость глубокая с указанием на отсутствие или слабую выраженность нарушения поведения, связанная с хромосомными нарушениями
  5. F78.81 Другие формы умственной отсталости с другими нарушениями поведения, обусловленные предшествующей инфекцией или интоксикацией
  6. F95.1 Хронические моторные тики или вокализмы.
  7. Homo ergaster, или Мальчик из Турканы.
  8. I Происхождение человека и цивилизации
  9. I.4. Библиографический поиск необходимых литературных источников
  10. II. Материалы судебной и иной юридической практики (если они есть в работе)
  11. II. Однородные члены предложения могут отделяться от обобщающего слова знаком тире (вместо обычного в таком случае двоеточия), если они выполняют функцию приложения со значением уточнения.
  12. II. Тип цикличного цивилизационного развития (восточный тип).


Последнее изменение этой страницы: 2016-03-17; Просмотров: 99;


lektsia.info 2017 год. Все права принадлежат их авторам! Главная