免责声明:即使您对此答案感到满意,您也应该等待,以防出现更好的答案,因为我远非专家,这些可能不是最好的方法。
第一种方法:
//two different sequences
struct A
{
MY_MACRO1((int)(float)(double),(x)(y)(z))
};
我认为这种方法给出了看起来不那么可怕的宏:
#define DECLARE_DATA_MEMBER1(R,TYPES,INDEX,NAME) \
BOOST_PP_SEQ_ELEM(INDEX,TYPES) BOOST_PP_CAT(m_,NAME);
#define DEFINE_ACCESSOR1(R,TYPES,INDEX,NAME) \
BOOST_PP_SEQ_ELEM(INDEX,TYPES) NAME(){ return BOOST_PP_CAT(m_,NAME); }
#define MY_MACRO1(TYPES,NAMES) \
BOOST_PP_SEQ_FOR_EACH_I(DECLARE_DATA_MEMBER1,TYPES,NAMES) \
public: \
BOOST_PP_SEQ_FOR_EACH_I(DEFINE_ACCESSOR1,TYPES,NAMES)
MY_MACRO
得到两个序列:TYPES
and NAMES
。为了声明数据成员,我使用BOOST_PP_SEQ_FOR_EACH_I
在序列上NAMES
使用宏DECLARE_DATA_MEMBER1
并具有序列TYPES
作为数据。这个“唤起”DECLARE_DATA_MEMBER1
有 4 个参数:R
这是未使用的(我不知道它是做什么的),TYPES
(类型的顺序),INDEX
(告诉我们现在处于哪个迭代中,从 0 开始),以及NAME
(原元素NAMES
与本次迭代相对应的序列)。
的“身体”DECLARE_DATA_MEMBER1
and DEFINE_ACCESSOR1
很简单,我们只需得到INDEX
类型序列中的第一个元素,并连接m_
with NAME
.
第二种方法:
//just one sequence but you need to put two sets of parentheses around each pair
struct B
{
MY_MACRO2(((int, x))((float,y))((double,z)))
};
这仍然相当简单,但有一个必须使用双括号的不方便之处。
#define DECLARE_DATA_MEMBER2(R,_,TYPE_AND_NAME) \
BOOST_PP_TUPLE_ELEM(2,0,TYPE_AND_NAME) BOOST_PP_CAT(m_,BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME));
#define DEFINE_ACCESSOR2(R,_,TYPE_AND_NAME) \
BOOST_PP_TUPLE_ELEM(2,0,TYPE_AND_NAME) BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME)(){ return BOOST_PP_CAT(m_,BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME)); }
#define MY_MACRO2(TYPES_AND_NAMES) \
BOOST_PP_SEQ_FOR_EACH(DECLARE_DATA_MEMBER2,_,TYPES_AND_NAMES) \
public: \
BOOST_PP_SEQ_FOR_EACH(DEFINE_ACCESSOR2,_,TYPES_AND_NAMES)
这次只有一个序列,因此我们不需要辅助宏中的索引。为此原因BOOST_PP_SEQ_FOR_EACH
使用宏用于 TYPES_AND_NAMESDECLARE_DATA_MEMBER2
并且不传递任何额外的数据。该宏接收三个“参数”:R
再次未使用,_
(or DATA
,此处也未使用),以及TYPE_AND_NAME
(形式为元组(TYPE,NAME)
).
在两个辅助宏的“主体”中BOOST_PP_TUPLE_ELEM
用于获取类型(索引=0)或名称(索引=1)。该宏需要传递元组的大小、所需元素的索引和元组。
第三种方法:
//one sequence but the macro is more complex
struct C
{
MY_MACRO3((int,x)(float,y)(double,z))
};
这个宏大量借鉴了BOOST_FUSION_ADAPT_STRUCT
和类似的宏。
//Heavily "inspired" from BOOST_FUSION_ADAPT_STRUCT
#define CREATE_MY_MACRO_PLACEHOLDER_FILLER_0(X, Y) \
((X, Y)) CREATE_MY_MACRO_PLACEHOLDER_FILLER_1
#define CREATE_MY_MACRO_PLACEHOLDER_FILLER_1(X, Y) \
((X, Y)) CREATE_MY_MACRO_PLACEHOLDER_FILLER_0
#define CREATE_MY_MACRO_PLACEHOLDER_FILLER_0_END
#define CREATE_MY_MACRO_PLACEHOLDER_FILLER_1_END
#define DECLARE_DATA_MEMBER3(R,_,TYPE_AND_NAME) \
BOOST_PP_TUPLE_ELEM(2,0,TYPE_AND_NAME) BOOST_PP_CAT(m_,BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME));
#define DEFINE_ACCESSOR3(R,_,TYPE_AND_NAME) \
BOOST_PP_TUPLE_ELEM(2,0,TYPE_AND_NAME) BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME)(){ return BOOST_PP_CAT(m_,BOOST_PP_TUPLE_ELEM(2,1,TYPE_AND_NAME)); }
#define MY_MACRO3(TYPES_AND_NAMES) \
BOOST_PP_SEQ_FOR_EACH(DECLARE_DATA_MEMBER3,_,BOOST_PP_CAT(CREATE_MY_MACRO_PLACEHOLDER_FILLER_0 TYPES_AND_NAMES,_END)) \
public: \
BOOST_PP_SEQ_FOR_EACH(DEFINE_ACCESSOR3,_,BOOST_PP_CAT(CREATE_MY_MACRO_PLACEHOLDER_FILLER_0 TYPES_AND_NAMES,_END))
在这种方法中,辅助宏基本上没有改变。唯一(大)的区别是 for_each 中使用的序列不仅仅是TYPES_AND_NAMES
but BOOST_PP_CAT(CREATE_MY_MACRO_PLACEHOLDER_FILLER_0 TYPES_AND_NAMES,_END)
。这是强制使用双括号的巧妙技巧。它的工作原理如下:
CREATE_MY_MACRO_PLACEHOLDER_FILLER_0(int,x)(float,y)_END
//CREATE_MY_MACRO_PLACEHOLDER_FILLER_0(A,B)->((A,B))CREATE_MY_MACRO_PLACEHOLDER_FILLER_1
((int,x))CREATE_MY_MACRO_PLACEHOLDER_FILLER_1(float,y)_END
//CREATE_MY_MACRO_PLACEHOLDER_FILLER_1(A,B)->((A,B))CREATE_MY_MACRO_PLACEHOLDER_FILLER_0
((int,x))((float,y))CREATE_MY_MACRO_PLACEHOLDER_FILLER_0_END
//CREATE_MY_MACRO_PLACEHOLDER_FILLER_0_END->
((int,x))((float,y))
在 Coliru 上运行。 http://coliru.stacked-crooked.com/a/1ac9bef8c1b6ed87