C++ variadic template function parameter with default value

前端 未结 2 1754
忘掉有多难
忘掉有多难 2021-02-07 02:30

I have a function which takes one parameter with a default value. Now I also want it to take a variable number of parameters and forward them to some other function. Function pa

相关标签:
2条回答
  • 2021-02-07 02:41

    Another approach would be to pass variadic arguments through a tuple.

    template <class... Args>
    void func (std::tuple<Args...> t, SomeSpecialType num = fromNum(5))
    {
      // don't forget to move t when you use it for the last time
    }
    

    Pros : interface is much simpler, overloading and adding default valued arguments is quite easy.

    Cons : caller has to manually wrap arguments in a std::make_tuple or std::forward_as_tuple call. Also, you'll probably have to resort to std::index_sequence tricks to implement the function.

    0 讨论(0)
  • 2021-02-07 03:00

    No, packs must be last.

    But you can fake it. You can detect what the last type in a pack is. If it is SomeSpecialType, you can run your func. If it isn't SomeSpecialType, you can recursively call yourself with your arguments forwarded and fromNum(5) appended.

    If you want to be fancy, this check can be done at compile time (ie, a different overload) using SFINAE techniques. But that probably isn't worth the hassle, considering that the "run-time" check will be constant on a given overload, and hence will almost certainly be optimized out, and SFINAE shouldn't be used lightly.

    This doesn't give you the signature you want, but it gives you the behavior you want. You'll have to explain the intended signature in comments.

    Something like this, after you remove typos and the like:

    // extract the last type in a pack.  The last type in a pack with no elements is
    // not a type:
    template<typename... Ts>
    struct last_type {};
    template<typename T0>
    struct last_type<T0> {
      typedef T0 type;
    };
    template<typename T0, typename T1, typename... Ts>
    struct last_type<T0, T1, Ts...>:last_type<T1, Ts...> {};
    
    // using aliases, because typename spam sucks:
    template<typename Ts...>
    using LastType = typename last_type<Ts...>::type;
    template<bool b, typename T=void>
    using EnableIf = typename std::enable_if<b, T>::type;
    template<typename T>
    using Decay = typename std::decay<T>::type;
    
    // the case where the last argument is SomeSpecialType:
    template<
      typename... Args,
      typename=EnableIf<
        std::is_same<
          Decay<LastType<Args...>>,
          SomeSpecialType
        >::value
      >
    void func( Args&&... args ) {
      // code
    }
    
    // the case where there is no SomeSpecialType last:    
    template<
      typename... Args,
      typename=EnableIf<
        !std::is_same<
          typename std::decay<LastType<Args...>>::type,
          SomeSpecialType
        >::value
      >
    void func( Args&&... args ) {
      func( std::forward<Args>(args)..., std::move(static_cast<SomeSpecialType>(fromNum(5))) );
    }
    
    // the 0-arg case, because both of the above require that there be an actual
    // last type:
    void func() {
      func( std::move(static_cast<SomeSpecialType>(fromNum(5))) );
    }
    

    or something much like that.

    0 讨论(0)
提交回复
热议问题