C ++ - Итерация над кортежом и разрешением постоянных параметров типа против

StackOverflow https://stackoverflow.com/questions/4832949

  •  27-10-2019
  •  | 
  •  

Вопрос

В настоящее время я нахожусь в процессе написания перегрузки арифметического оператора для кортежей. Оператор итерация над кортежом для выполнения операции на каждом из его отдельных элементов. Вот определение для оператора +=:

template< typename... Ts, std::size_t I = 0 >
inline typename std::enable_if< I == sizeof... (Ts), std::tuple< Ts... >& >::type operator +=(std::tuple< Ts... >& lhs, const std::tuple< Ts... >& rhs)
{
    return lhs;
}

template< typename... Ts, std::size_t I = 0 >
inline typename std::enable_if< I != sizeof... (Ts), std::tuple< Ts... >& >::type operator +=(std::tuple< Ts... >& lhs, const std::tuple< Ts... >& rhs)
{
    std::get< I >(lhs) += std::get< I >(rhs);
    return operator +=< Ts..., I + 1 >(lhs, rhs);
}

К сожалению, когда я пытаюсь позвонить оператору, GCC 4.6 не может решить, какую перегрузку он должен использовать. Например:

std::tuple< int, int, int, int > a = std::make_tuple(1, 2, 3, 4), b = std::make_tuple(5, 6, 7, 8);
a += b;

Дает следующую ошибку:

:/Workspace/raster/main.cpp:833:7:   instantiated from here
C:/Workspace/raster/main.cpp:809:45: error: no matching function for call to 'operator+=(std::tuple<int, int, int, int>&, const std::tuple<int, int, int, int>&)'
C:/Workspace/raster/main.cpp:809:45: note: candidates are:
C:/Workspace/raster/main.cpp:800:151: note: template<class ... Ts, unsigned int I> typename std::enable_if<(I == sizeof (Ts ...)), std::tuple<_TElements ...>&>::type operator+=(std::tuple<_TElements ...>&, const std::tuple<_TElements ...>&)
C:/Workspace/raster/main.cpp:806:83: note: template<class ... Ts, unsigned int I> typename std::enable_if<(I != sizeof (Ts ...)), std::tuple<_TElements ...>&>::type operator+=(std::tuple<_TElements ...>&, const std::tuple<_TElements ...>&)

Что странно, так как std::enable_if Условие должно отклонить неуместный вызов. На данный момент у меня есть следующий обходной путь, который на самом деле был моей предыдущей реализацией. Вышеуказанная версия на самом деле является попыткой упрощения.

template< std::size_t I, typename... Ts >
inline typename std::enable_if< I == sizeof... (Ts), std::tuple< Ts... >& >::type assignadd_impl(std::tuple< Ts... >& lhs, const std::tuple< Ts... >& rhs)
{
    return lhs;
}

template< std::size_t I, typename... Ts >
inline typename std::enable_if< I != sizeof... (Ts), std::tuple< Ts... >& >::type assignadd_impl(std::tuple< Ts... >& lhs, const std::tuple< Ts... >& rhs)
{
    std::get< I >(lhs) += std::get< I >(rhs);
    return assignadd_impl< I + 1, Ts... >(lhs, rhs);
}

template< typename... Ts >
inline std::tuple< Ts... >& operator +=(std::tuple< Ts... >& lhs, const std::tuple< Ts... >& rhs)
{
    return assignadd_impl< 0, Ts... >(lhs, rhs);
}

Это компилирует и работает, как и ожидалось. Почему упрощенная версия отказывается компилировать? Спасибо.

Это было полезно?

Решение

Использование явных указанных аргументов шаблона для функции или шаблона класса требует, чтобы любые пакеты параметров шаблона появлялись в конце общего списка параметров шаблона. Движущийся Ts... К концу списка параметров шаблона и соответствующее изменение вызовов заставляет код работать. Раздел 14.8.2.1 OF текущий черновик C ++ 0x утверждает, что пакеты параметров, которые не находятся в конце списка параметров шаблонов operator+= Во всех случаях по -прежнему вызывает ошибку Sfinae. Предыдущий вопрос имеет ссылку на точный текст, запрещающий его; Документация IBM говорит, что это тоже ошибка.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top