试图理解c中的宏定义

Ahm*_*yed 2 c macros

这是我试图理解的代码的一部分,但我无法理解这样的宏是如何工作的。第一行#define FOREACH_OMPD_STATE(macro)定义了一个类似 function_like 的宏,它有一个名为“ macro ”的参数,但是定义的其余部分是什么意思?

#define FOREACH_OMPD_STATE(macro)                                             \
                                                                              \
    /* first available state */                                               \
    macro (ompt_state_undefined, 0x102)      /* undefined thread state */     \
                                                                              \
    /* work states (0..15) */                                                 \
    macro (ompt_state_work_serial, 0x000)    /* working outside parallel */   \
    macro (ompt_state_work_parallel, 0x001)  /* working within parallel */    \
    macro (ompt_state_work_reduction, 0x002) /* performing a reduction */    
Run Code Online (Sandbox Code Playgroud)

这些不同的值是名为宏的参数还是什么?

Eri*_*hil 5

FOREACH_OMPD_STATE是一个X 宏。它提供了一个列表,通过传递各种宏来对列表项执行操作。例如,我们可以使用仅生成名称的宏来创建标识符列表enum

#define NameOnly(name, value)   name,

enum { FOREACH_OMPD_STATE(NameOnly) };
Run Code Online (Sandbox Code Playgroud)

该代码生成(添加行格式以提高可读性):

enum {
    ompt_state_undefined,
    ompt_state_work_serial,
    ompt_state_work_parallel,
    ompt_state_work_reduction,
};
Run Code Online (Sandbox Code Playgroud)

完成此操作后,我们可以使用另一个宏来创建数组初始值设定项,以使用宏enum提供的值填充由标识符索引的数组FOREACH_OMPD_STATE

#define MakeInitializer(name, value)    [name] = value,

int ArrayOfValues[] = { FOREACH_OMPD_STATE(MakeInitializer) };
Run Code Online (Sandbox Code Playgroud)

这会生成(为了可读性而添加的行格式):

int ArrayOfValues = {
    [ompt_state_undefined] = 0x102,
    [ompt_state_work_serial] = 0x000,
    [ompt_state_work_parallel] = 0x001,
    [ompt_state_work_reduction] = 0x002,
};
Run Code Online (Sandbox Code Playgroud)

关键思想是我们只需要在源代码中、定义宏时列出一次名称和值FOREACH_OMPD_STATE,然后就可以在以后以多种方式使用它们,而无需重复它们。