#define max(A, B) ((A) › (B) ? (A) : (B))
Хотя обращения к
x = max(p+q, r+s);
будет заменена на строку
x = ((p+q) › (r+s) ? (p+q) : (r+s));
Поскольку аргументы допускают любой вид замены, указанное определение
Если вы внимательно проанализируете работу
max(i++, j++) /* НЕВЕРНО */
вызовет увеличение
#define square(x) x*x /* НЕВЕРНО */
вызвать
Тем не менее макросредства имеют свои достоинства. Практическим примером их использования является частое применение
#undef getchar
int getchar(void) {…}
Как правило, это делается, чтобы заменить макроопределение настоящей функцией с тем же именем.
Имена формальных параметров не заменяются, если встречаются в заключенных в кавычки строках. Однако, если в замещающем тексте перед формальным параметром стоит знак #, этот параметр будет заменен на аргумент, заключенный в кавычки. Это может сочетаться с конкатенацией (склеиванием) строк, например, чтобы создать макрос отладочного вывода:
#define dprint(expr) printf(#expr " = %g\n", expr)
Обращение к
dprint(x/y);
развернется в
printf("x/y" " = %g\n", x/y);
а в результате конкатенации двух соседних строк получим
printf("x/y=%g\n", x/y);
Внутри фактического аргумента каждый знак " заменяется на \", а каждая \ на \\, так что результат подстановки приводит к правильной символьной константе.
Оператор ## позволяет в макрорасширениях конкатенировать аргументы. Если в замещающем тексте параметр соседствует с ##, то он заменяется соответствующим ему аргументом, а оператор ## и окружающие его символы-разделители выбрасываются. Например, в макроопределении
#define paste(front, back) front ## back
так что
Правила вложенных использований оператора ## не определены; другие подробности, относящиеся к ##, можно найти в приложении A.
Упражнение 4.14. Определите swap(t,x,y) в виде макроса, который осуществляет обмен значениями указанного типа t между аргументами x и y. (Примените блочную структуру.)
4.11.3 Условная компиляция
Самим ходом препроцессирования можно управлять с помощью условных инструкций. Они представляют собой средство для выборочного включения того или иного текста программы в зависимости от значения условия, вычисляемого вовремя компиляции.
Вычисляется константное целое выражение, заданное в строке #if. Это выражение не должно содержать ни одного оператора sizeof или приведения к типу и ни одной enum-константы. Если оно имеет ненулевое значение, то будут включены все последующие строки вплоть до #endif, или #elif, или #else. (Инструкция препроцессора #elif похожа на else if.) Выражение defined(
Например, чтобы застраховаться от повторного включения заголовочного файла
#if !defined(HDR)
#define HDR
/* здесь содержимое hdr.h */
#endif
При первом включении файла
Вот пример цепочки проверок имени
#if SYSTEM == SYSV
#define HDR "sysv.h"