В C ++ 11 нет способа хранить временную строку где-либо во время компиляции. Поэтому я могу предложить вам такой подход:
(Это быстро сделанный набросок, но хорошо описательный)
#include <stdio.h>
template <char...>
struct StringTuple;
template <char TargHead>
struct StringTuple<TargHead> {
static constexpr char kSymbol = TargHead;
static void print() {
printf(kSymbol ? "%c\n" : "\n", kSymbol);
}
};
template <char TargHead, char... TargTail>
struct StringTuple<TargHead, TargTail...> {
using Next = StringTuple<TargTail...>;
static constexpr char kSymbol = TargHead;
static void print() {
if (kSymbol) {
printf("%c", kSymbol);
Next::print();
} else {
printf("\n");
}
}
};
constexpr int length(char *string) {
return (string[0] == 0) ? 1 : (length(string + 1) + 1);
}
constexpr char get(char *string, int i) {
return i < length(string) ? string[i] : 0;
}
#define ST(string) \
StringTuple< \
get(string, 0), \
get(string, 1), \
get(string, 2), \
get(string, 3), \
get(string, 4), \
get(string, 5), \
get(string, 6), \
get(string, 7), \
get(string, 8), \
get(string, 9), \
get(string, 10), \
get(string, 11), \
get(string, 12), \
get(string, 13), \
get(string, 14), \
get(string, 15), \
get(string, 16), \
get(string, 17), \
get(string, 18), \
get(string, 19), \
get(string, 20), \
get(string, 21), \
get(string, 22), \
get(string, 23), \
get(string, 24), \
get(string, 25), \
get(string, 26), \
get(string, 27), \
get(string, 28), \
get(string, 29), \
get(string, 30), \
get(string, 31), \
get(string, 32), \
get(string, 33), \
get(string, 34), \
get(string, 35), \
get(string, 36), \
get(string, 37), \
get(string, 38), \
get(string, 39), \
get(string, 40), \
get(string, 41), \
get(string, 42) \
>
int main() {
ST("Hello, compile-time world!")::print();
}
Код Bash для генерации части макроса:
for i in `seq 0 42`; do echo " get(string, $i), \\"; done
Вы должны передать большое число (1000 или более) этому генератору для поддержки всех ваших строк, и вы должны сделать статическое утверждение, если строка превышает этот предел.
Я использую такие сгенерированные макросы в своих научных проектах. Я знаю, это кажется грязным, но это работает. Пример использования сгенерированного макроса:
#define PRINT(a) print(a);
FOREACH_MACRO(PRINT, a, b, c) // print(a);print(b);print(c);
Я постараюсь найти более симпатичное решение, но сначала я воспользуюсь этим.