c/c++:提取可變引數巨集__VA_ARGS__中偶數位置引數
阿新 • • 發佈:2019-02-03
考慮一個可變引數巨集__VA_ARGS__
中奇數位代表引數型別,偶數位代表引數名,想要提取__VA_ARGS__
中所有的偶數位的引數名,該怎麼實現呢?
利用上一篇部落格《c/c++:計算可變引數巨集 __VA_ARGS__
的引數個數》的成果可以獲取__VA_ARGS__
中引數的個數。在這個基礎上新增一系列巨集定義就可以實現,下面是完整程式碼及測試用例, gcc下測試通過:
// 計算 __VA_ARGS__ 引數個數,最大支援64個引數
#define FL_ARG_COUNT(...) FL_INTERNAL_ARG_COUNT_PRIVATE(0, ##__VA_ARGS__,\
64 , 63, 62, 61, 60, \
59, 58, 57, 56, 55, 54, 53, 52, 51, 50, \
49, 48, 47, 46, 45, 44, 43, 42, 41, 40, \
39, 38, 37, 36, 35, 34, 33, 32, 31, 30, \
29, 28, 27, 26, 25, 24, 23, 22, 21, 20, \
19, 18, 17, 16, 15, 14, 13, 12, 11, 10, \
9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
#define FL_INTERNAL_ARG_COUNT_PRIVATE(\
_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, \
_10, _11, _12, _13, _14, _15, _16, _17, _18, _19, \
_20, _21, _22, _23, _24, _25, _26, _27, _28, _29, \
_30, _31, _32, _33, _34, _35, _36, _37, _38, _39, \
_40, _41, _42, _43, _44, _45, _46, _47, _48, _49, \
_50, _51, _52, _53, _54, _55, _56, _57, _58, _59, \
_60, _61, _62, _63, _64, N, ...) N
// 引數拼接
#define FL_CONCAT_(l,r) l ## r
#define FL_CONCAT(l,r) FL_CONCAT_(l,r)
#define FL_ARG0()
#define FL_ARG2(t,v) v
#define FL_ARG4(t,v,...) FL_ARG2(t,v),FL_ARG2(__VA_ARGS__)
#define FL_ARG6(t,v,...) FL_ARG2(t,v),FL_ARG4(__VA_ARGS__)
#define FL_ARG8(t,v,...) FL_ARG2(t,v),FL_ARG6(__VA_ARGS__)
#define FL_ARG10(t,v,...) FL_ARG2(t,v),FL_ARG8(__VA_ARGS__)
#define FL_ARG12(t,v,...) FL_ARG2(t,v),FL_ARG10(__VA_ARGS__)
#define FL_ARG14(t,v,...) FL_ARG2(t,v),FL_ARG12(__VA_ARGS__)
#define FL_ARG16(t,v,...) FL_ARG2(t,v),FL_ARG14(__VA_ARGS__)
#define FL_ARG18(t,v,...) FL_ARG2(t,v),FL_ARG16(__VA_ARGS__)
#define FL_ARG20(t,v,...) FL_ARG2(t,v),FL_ARG18(__VA_ARGS__)
#define FL_ARG22(t,v,...) FL_ARG2(t,v),FL_ARG20(__VA_ARGS__)
#define FL_ARG24(t,v,...) FL_ARG2(t,v),FL_ARG22(__VA_ARGS__)
#define FL_ARG26(t,v,...) FL_ARG2(t,v),FL_ARG24(__VA_ARGS__)
#define FL_ARG28(t,v,...) FL_ARG2(t,v),FL_ARG26(__VA_ARGS__)
#define FL_ARG30(t,v,...) FL_ARG2(t,v),FL_ARG28(__VA_ARGS__)
#define FL_ARG32(t,v,...) FL_ARG2(t,v),FL_ARG30(__VA_ARGS__)
#define FL_ARG34(t,v,...) FL_ARG2(t,v),FL_ARG32(__VA_ARGS__)
#define FL_ARG36(t,v,...) FL_ARG2(t,v),FL_ARG34(__VA_ARGS__)
#define FL_ARG38(t,v,...) FL_ARG2(t,v),FL_ARG36(__VA_ARGS__)
#define FL_ARG40(t,v,...) FL_ARG2(t,v),FL_ARG38(__VA_ARGS__)
#define FL_ARG42(t,v,...) FL_ARG2(t,v),FL_ARG40(__VA_ARGS__)
#define FL_ARG44(t,v,...) FL_ARG2(t,v),FL_ARG42(__VA_ARGS__)
#define FL_ARG46(t,v,...) FL_ARG2(t,v),FL_ARG44(__VA_ARGS__)
#define FL_ARG48(t,v,...) FL_ARG2(t,v),FL_ARG46(__VA_ARGS__)
#define FL_ARG50(t,v,...) FL_ARG2(t,v),FL_ARG48(__VA_ARGS__)
#define FL_ARG52(t,v,...) FL_ARG2(t,v),FL_ARG50(__VA_ARGS__)
#define FL_ARG54(t,v,...) FL_ARG2(t,v),FL_ARG52(__VA_ARGS__)
#define FL_ARG56(t,v,...) FL_ARG2(t,v),FL_ARG54(__VA_ARGS__)
#define FL_ARG58(t,v,...) FL_ARG2(t,v),FL_ARG56(__VA_ARGS__)
#define FL_ARG60(t,v,...) FL_ARG2(t,v),FL_ARG58(__VA_ARGS__)
#define FL_ARG62(t,v,...) FL_ARG2(t,v),FL_ARG60(__VA_ARGS__)
#define FL_ARG64(t,v,...) FL_ARG2(t,v),FL_ARG62(__VA_ARGS__)
// 提取動態引數表中的偶數位引數,比如 一個引數序列:1,2,3,4,返回 2,4,最大支援64個引數
// 引數個數為奇數時會導致編譯報錯
#define FL_EVEN_ARG_(...) FL_CONCAT(FL_ARG,FL_ARG_COUNT(__VA_ARGS__))(__VA_ARGS__)
#define FL_EVEN_ARG(...) FL_EVEN_ARG_(__VA_ARGS__)
/* test case */
static const int a0[] = {FL_EVEN_ARG()}; // 返回空
static const int a1[] ={FL_EVEN_ARG(1,2)}; // 2
static const int a4[] ={FL_EVEN_ARG(1,2,3,4)}; // 2,4
static const int a20[] ={FL_EVEN_ARG(1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20)}; // 2,4,6,8,10,12,14,16,18,20