How to extract __VA_ARGS__?

后端 未结 3 1207
轮回少年
轮回少年 2021-01-06 04:32

I hava a macro to call static function for each args.

For example:

#define FOO(X) X::do();
#define FOO_1(X,Y) X::do(); Y::do();

My

3条回答
  •  太阳男子
    2021-01-06 05:28

    Actually you can partially workaround this.
    You can directly and freely extract every member of neither __VA_ARGS__ nor variadic templates of C++11. But you can have the very first element. For example let's say we have a macro named OUT(...) and we want to produce std::cout << A << B << C ... where A, B, C are the variadic arguments of macro. Try this:

    #include 
    
    #define SEPERATOR <<
    #define GET_1_OF(element1, ...) element1
    #define GET_2_OF(element1, ...) element1 SEPERATOR GET_1_OF(__VA_ARGS__) 
    #define GET_3_OF(element1, ...) element1 SEPERATOR GET_2_OF(__VA_ARGS__) 
    #define BAR(...) GET_3_OF(__VA_ARGS__)
    int main()
    {
        std::cout << BAR(1,2,3,4,5);
        return 0;
    }
    

    This is of course not the solution you are after. But you can augment the number of GET_N_OF to do what you want. Note that SEPERATOR is << so that we MACRO can write 1 << 2 << 3 and so on.
    Now, we have a problem in this code. Please change BAR(1,2,3,4,5) with BAR(1) You will see that it is giving an error. This is because it was expecting 3 arguments, although it is not problem to have more arguments (because it is variadic) we are having extra SEPERATOR. So in order to solve this problem instead of using BAR(...) use GET_N_OF(...) (since you know the number of arguments):

    #include 
    
    #define SEPERATOR <<
    #define GET_1_OF(element1, ...) element1
    #define GET_2_OF(element1, ...) element1 SEPERATOR GET_1_OF(__VA_ARGS__) 
    #define GET_3_OF(element1, ...) element1 SEPERATOR GET_2_OF(__VA_ARGS__) 
    #define GET_4_OF(element1, ...) element1 SEPERATOR GET_3_OF(__VA_ARGS__) 
    #define GET_5_OF(element1, ...) element1 SEPERATOR GET_4_OF(__VA_ARGS__) 
    
    int main()
    {
        std::cout << GET_5_OF(1,2,3,4,5);
        std::cout << GET_1_OF(1);
        return 0;
    }
    

    Please note that if you do not know what you are doing do not use MACROs at all! My response was just to share fun MACRO code that may be beneficial for you. I always discourage the usage of MACROs until they are remarkably necessary.

提交回复
热议问题