Общее программирование с концепциями, конфликтующими с конструктором копирования

Я пытаюсь использовать общий стиль кодирования, чтобы обернуть API ограниченной концепции в общие стеки. Проблема, с которой я сталкиваюсь, состоит в том, что один из моих конструкторов, самый важный, который использует шаблонный стек, переопределяет мой конструктор копирования. У меня есть суть рассматриваемого класса, некоторый тестовый код и ошибка, которую я получаю от компилятора здесь: https://gist.github.com/biot023/5178831

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

Может ли кто-нибудь посоветовать, как я мог бы явно вызвать конструктор копирования (что мне действительно нужно сделать в моем тестовом коде), или, может быть, показать мне, где я ошибаюсь?

1 ответ

Решение

Проблема в том, что ваш шаблон конструктора лучше соответствует вашему конструктору копирования:

// Copy constructor
WrappedStack( const WrappedStack &other )

// Constructor template
template <typename S>
WrappedStack( S &stack )

Когда создается экземпляр объекта типа WrappedStack<T> (где T является параметром шаблона класса), экземпляр сигнатуры шаблона конструктора будет выглядеть так:

WrappedStack( WrappedStack<T> &stack )

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

Чтобы заставить компилятор использовать конструктор копирования, когда это необходимо, вы можете использовать SFINAE, чтобы вызвать ошибку замещения при создании экземпляра шаблона конструктора с объектом типа WrappedStack<T>или типа, который неявно преобразуется в WrappedStack<T>, Например:

#include <type_traits>

template<
    typename S,
    typename std::enable_if<
        !std::is_convertible<S, WrappedStack<T> const&>::value
        >::type* = nullptr>
WrappedStack( S &stack ) : __stack( new wrapped_stack_t<S, T>( stack ) ) {}

Посмотрите живой пример этого успешного компиляции.

Другие вопросы по тегам