Используйте SFINAE, чтобы включить частичную специализацию, основанную на размере пакета - PullRequest
0 голосов
/ 15 февраля 2019

Я пытаюсь сделать шаблон, который работает с символами, во время компиляции.В этом случае я хочу наложить ограничение на то, что всегда должно быть точное кратное определенному количеству символов.

В случае, когда нет точного соответствия, я хочу дополнить их 0 наГлава пакета.

(Помимо этого, мотивация состоит в том, чтобы (во время компиляции, как часть более серьезной проблемы) добавить поддержку отображения двоичных и шестнадцатеричных литералов в std::array<unsigned char, N>, это работаеткрасиво, за исключением разметки вещей, которые не кратны байту).

Вот упрощенный пример того, что я пытаюсь сделать, чтобы заставить отступ работать:

// Thingy operates on N*4 chars - if that's not met use inheritance to 0 pad until it is met.
template<char ...Args>
struct thingy : thingy<0, Args...> {
    // All we do here is recursively add one more 0 via inheritance until the N*4 rule is met
};

// This specialisation does the real work, N=1 case only
template<char a, char b, char c, char d>
struct thingy<a,b,c,d> {
    enum { value = (a << 24) | (b << 16) | (c << 8) | d }; 
};

// This handles chunking the N*4 things into N cases of work. Does work along the way, only allowed for exact N*4 after padding has happened.
template <char a, char b, char c, char d, char ...Args>
struct thingy<a,b,c,d,Args...> : thingy<a,b,c,d> {
    static_assert(sizeof...(Args) % 4 == 0); // PROBLEM: this is a we're a better match than the template that pads things, how do we stop that?
    // Do something with the value we just got and/or the tail as needed
    typedef thingy<a,b,c,d> head;
    typedef thingy<Args...> tail;
};

int main() {
    thingy<1,1,1,1,1>(); // This should be equivalent to writing thingy<0,0,0,1,1,1,1,1>()
}

Это поражает мой static_assert.Проблема в том, что мы всегда совпадаем с неверной специализацией, которую я ожидал, потому что она более специализированная.


Поэтому я оглянулся и нашел несколько примеров той же проблемы для функции , но, насколько я вижу, ни одно из них здесь не применимо.

Я попробовал еще несколько вещей, ни одна из которых не работала так, как я надеялся, сначала был наивно enable_if на sizeof...(Args) именно там, где я хотел:

template <char a, char b, char c, char d, typename std::enable_if<sizeof...(Args) % 4 == 0, char>::type ...Args>
struct thingy<a,b,c,d,Args...> : thingy<a,b,c,d> {
    // ...
};

Это незаконно, насколько я могу судить, и, конечно, не работает с моими компиляторами - в момент, когда нам нужно запросить sizeof...(Args)Args еще не существует.

Мы не можем юридически добавить еще один аргумент шаблона после пакета, насколько я могу судить, это также не удалось:

template <char a, char b, char c, char d, char ...Args, typename std::enable_if<sizeof...(Args) % 4 == 0, int>::type=0>
struct thingy<a,b,c,d,Args...> : thingy<a,b,c,d> {
    // ...
};

сошибка:

pad_params_try3.cc:17:8: error: default template arguments may not be used in partial specializations

Я также пробовал SFINAE в самом наследовании, но это не является законным местом для этого:

template <char a, char b, char c, char d, char ...Args>
struct thingy<a,b,c,d,Args...> : std::enable_if<sizeof...(Args) % 4 == 0, thingy<a,b,c,d>>::type {
    // ...
};

В этом мы попадаем в обаstatic_assert и сбой, который является ошибкой в ​​enable_if.

pad_params_try4.cc:17:8: error: no type named 'type' in 'struct std::enable_if<false, thingy<'\001', '\001', '\001', '\001'> >'
 struct thingy<a,b,c,d,Args...> : std::enable_if<sizeof...(Args) % 4 == 0, thingy<a,b,c,d>>::type {
        ^~~~~~~~~~~~~~~~~~~~~~~
pad_params_try4.cc:18:5: error: static assertion failed
     static_assert(sizeof...(Args) % 4 == 0);

Насколько я могу судить по прочтению чуть больше , это даже можно считать дефектом , но это не сильно мне помогает прямо сейчас.

Как я могу обойти это, с тем, что у меня есть в C ++ 14, gcc 6.x?Есть ли более простой вариант, чем полностью вернуться к чертежной доске?

Ответы [ 4 ]

0 голосов
/ 15 февраля 2019

Я бы избавился от списка с головой / хвостом и использовал бы std::tuple, в результате чего:

// No variadic here
template <char a, char b, char c, char d>
struct thingy {
    enum { value = (a << 24) | (b << 16) | (c << 8) | d }; 
};

template <typename Seq, char... Cs>
struct thingies_impl;

template <std::size_t ...Is, char... Cs>
struct thingies_impl<std::index_sequence<Is...>, Cs...>
{
private:
    static constexpr char get(std::size_t n)
    {
        constexpr char cs[] = {Cs...};
        constexpr std::size_t paddingSize = (4 - (sizeof...(Cs) % 4)) % 4;
        return (n < paddingSize) ? '\0' : cs[n - paddingSize];
    }

public:
    using type = std::tuple<thingy<get(4 * Is),
                                   get(4 * Is + 1),
                                   get(4 * Is + 2),
                                   get(4 * Is + 3)>...>;  
};

template <char... Cs>
using thingies = thingies_impl<std::make_index_sequence<(sizeof...(Cs) + 3) / 4>, Cs...>;

Демо

0 голосов
/ 15 февраля 2019

Прежде всего, простое решение в C ++ 17, использующее рекурсивную вспомогательную функцию if constexpr для выполнения заполнения для вас:

template<char ... Args>
auto getThingyPadded()
{
    if constexpr (sizeof...(Args) % 4 != 0)
        return getThingyPadded<0, Args...>();
    else
        return thingy<Args...>{};
}

Чтобы сделать этот C ++ 14, нам нужно использоватьСФИНАЕ вместо if constexpr.Мы можем добавить вызов, который вычисляет sizeof...(Args), чтобы обойти описанные вами проблемы:

template<bool B, class U = void>
using enableIfT = typename std::enable_if<B, U>::type;

template<std::size_t N, enableIfT<(N % 4 == 0)>* = nullptr, char ... Args>
auto getThingyPaddedHelper()
{
    return thingy<Args...>{};
}

template<std::size_t N, enableIfT<(N % 4 != 0)>* = nullptr, char ... Args>
auto getThingyPaddedHelper()
{
    return getThingyPaddedHelper<N+1, nullptr, 0, Args...>();
}

template<char ... Args>
auto getThingyPadded()
{
    return getThingyPaddedHelper<sizeof...(Args), nullptr, Args...>();
}

Демо!

0 голосов
/ 15 февраля 2019

Еще один подход множественного наследования (с исправлением и упрощением из Jarod42 (спасибо!)).

#include <utility>

template <char a, char b, char c, char d, char ... Args>
struct t_base : public t_base<Args...>
 {
   typedef t_base<a,b,c,d> head;
   typedef t_base<Args...> tail;
 };

template <char a, char b, char c, char d>
struct t_base<a, b, c, d>
 { enum { value = (a << 24) | (b << 16) | (c << 8) | d }; };


template <typename, typename>
struct t_helper;

template <std::size_t ... Is, char ... Cs>
struct t_helper<std::index_sequence<Is...>,
                std::integer_sequence<char, Cs...>>
   : public t_base<(Is, '0')..., Cs...>
 { };


template <char ... Cs>
struct thingy :
   public t_helper<std::make_index_sequence<(4u - sizeof...(Cs) % 4u) % 4u>,
                   std::integer_sequence<char, Cs...>>
 { };


int main ()
 {
 }
0 голосов
/ 15 февраля 2019

Как насчет немного другого подхода с множественным наследованием и промежуточной вспомогательной структурой, которая выполняет заполнение?

// This handles chunking the N*4 things into N cases of work. Does work along the way, only allowed for exact N*4 after padding has happened.
template <char a, char b, char c, char d, char... Args>
struct thingy_impl : thingy_impl<a, b, c, d>, thingy_impl<Args...> {
    static_assert(sizeof...(Args) % 4 == 0);
    // Do something with the value we just got and/or the tail as needed
    typedef thingy_impl<a,b,c,d> head;
    typedef thingy_impl<Args...> tail;
};

template<char a, char b, char c, char d>
struct thingy_impl<a,b,c,d> {
    enum { value = (a << 24) | (b << 16) | (c << 8) | d }; 
};

template<int REMAINDER, char... Args>
struct padding;

template<char... Args>
struct padding<0,Args...> { using type = thingy_impl<Args...>; };

template<char... Args>
struct padding<1,Args...> { using type = thingy_impl<0,0,0,Args...>; };

template<char... Args>
struct padding<2,Args...> { using type = thingy_impl<0,0,Args...>; };

template<char... Args>
struct padding<3,Args...> { using type = thingy_impl<0,Args...>; };

template<char... Args>
struct thingy : padding<sizeof...(Args) % 4, Args...>::type { };

int main() {
    thingy<1,1,1,1,1>(); // This should be equivalent to writing thingy<0,0,0,1,1,1,1,1>()
}

Демонстрация с диагностическим выводом.

...