Search code examples
c++tuplesvariadic-templatesstdarraynon-type-template-parameter

variadic template, Range, C++


Consider I have a struct.

template<typename T>
struct Range{
    T min, max;
    
    Range() = default;
    Range(T min = std::numeric_limits<T>::min(),T max = std::numeric_limits::max()) :
                                   min(min), max(max){
    } 
};

So this struct holds a range for any type T, and if not given it the range becomes the numeric limits of that type. Note: Of course I check for T to be fundamental type but this question is not about that.

Now in my use case I want to have a struct like this

template<typename T,Range<T>...ranges>
struct Something{
    static std::array<Range<T>,sizeof...(ranges)> RANGES = {ranges...};
};

But when I want to "define" that something with this using Defined = Something<int,Range<int>(100,200),Range<int>(200,300)> I get compile-time error, because Range<T> is non-type template parameter and I am not using C++ 20. So I went further and updated my range struct like this.

template<typename T, T MIN = std::numeric_limits<T>::min(), T MAX = std::numeric_limits<T>::max()>
struct Range {
    T min = MIN, max = MAX;

    Range(T min, T max) : min(min), max(max) {}

    Range() = default;
};

And my Something struct's template becomes this typename<T,typename... Args> so now using Defined = Something<int,Range<int,100,200>,Range<int,200,300>> works. So at this point I've got 2 questions, how can I statically assert for each of args to be a Range<T,T min, T max> . The second questions is about storing that. Look, I know that I have to store ...Args in a tuple, but if you look at my implementation of Range, can I somehow wrap them in an std::array<Range<T>,sizeof(args)>?


Solution

  • You can create trait is_range:

    template <typename> struct is_range : std::false_type{};
    
    template<typename T, T MIN, T MAX>
    struct is_range<Range<T, MIN, MAX> : std::true_type{};
    

    and then fold expression (C++17)

    template <typename ...Ts>
    struct Something{
        static_assert((is_range<Ts>::value && ...));
        static std::tuple<Ts...> RANGES;
    };