Строка макроопределения
У меня возникла проблема - мне нужно использовать значение макроса как в виде строки, так и как целое.
#define RECORDS_PER_PAGE 10
/*... */
#define REQUEST_RECORDS \
"SELECT Fields FROM Table WHERE Conditions" \
" OFFSET %d * " #RECORDS_PER_PAGE \
" LIMIT " #RECORDS_PER_PAGE ";"
char result_buffer[RECORDS_PER_PAGE][MAX_RECORD_LEN];
/* ...and some more uses of RECORDS_PER_PAGE, elsewhere... */
Это не удается с сообщением о "stray #", и даже если это сработало, я предполагаю, что имена макросов будут стробированы, а не значения. Конечно, я могу передать значения окончательному методу ("LIMIT %d ", page*RECORDS_PER_PAGE
), но он не является ни хорошеньким, ни эффективным.
Это случается так, когда я хочу, чтобы препроцессор не обрабатывал строки особым образом и обрабатывал их содержимое, как обычный код.
На данный момент я сделал это с помощью #define RECORDS_PER_PAGE_TXT "10"
, но, понятно, я не доволен этим.
Как это сделать правильно?
Ответы
Ответ 1
Макрос xstr
, определенный ниже, будет сжиматься после выполнения макрообмена.
#define xstr(a) str(a)
#define str(a) #a
#define RECORDS_PER_PAGE 10
#define REQUEST_RECORDS \
"SELECT Fields FROM Table WHERE Conditions" \
" OFFSET %d * " xstr(RECORDS_PER_PAGE) \
" LIMIT " xstr(RECORDS_PER_PAGE) ";"
Ответ 2
Попробуйте выполнить двойное экранирование кавычек
#define RECORDS_PER_PAGE 10
#define MAX_RECORD_LEN 10
/*... */
#define DOUBLEESCAPE(a) #a
#define ESCAPEQUOTE(a) DOUBLEESCAPE(a)
#define REQUEST_RECORDS \
"SELECT Fields FROM Table WHERE Conditions" \
" OFFSET %d * " ESCAPEQUOTE(RECORDS_PER_PAGE) \
" LIMIT " ESCAPEQUOTE(RECORDS_PER_PAGE) ";"
char result_buffer[RECORDS_PER_PAGE][MAX_RECORD_LEN];
int main(){
char * a = REQUEST_RECORDS;
}
компилируется для меня. Маркер RECORDS_PER_PAGE
будет расширяться макрокомандом ESCAPEQUOTE
, который затем отправляется в DOUBLEESCAPE
для цитирования.
Ответ 3
#include <stdio.h>
#define RECORDS_PER_PAGE 10
#define TEXTIFY(A) #A
#define _REQUEST_RECORDS(OFFSET, LIMIT) \
"SELECT Fields FROM Table WHERE Conditions" \
" OFFSET %d * " TEXTIFY(OFFSET) \
" LIMIT " TEXTIFY(LIMIT) ";"
#define REQUEST_RECORDS _REQUEST_RECORDS(RECORDS_PER_PAGE, RECORDS_PER_PAGE)
int main() {
printf("%s\n", REQUEST_RECORDS);
return 0;
}
Выходы:
SELECT Fields FROM Table WHERE Conditions OFFSET %d * 10 LIMIT 10;
Обратите внимание на косвенность на _REQUEST_RECORDS, чтобы оценить аргументы перед их изложением.