如何利用模板复制和移动构造函数和赋值运算符?

时间:2019-01-05 10:02:10

标签: c++ templates copy-constructor assignment-operator default-constructor

考虑以下C ++代码,但我尝试避免preference of non-template copy&move constructors and assignment operators失败:

template<typename T> class A {
public:
    A() { /* implementation here */ }

    // Remove from the overloads the default copy&move constructors and assignment operators
    A(const A&) = delete;
    A& operator=(const A&) = delete;
    A(A&&) = delete;
    A& operator=(A&&) = delete;

    // I want these to be used e.g. by std::vector
    template<typename U> A(const A<U>& fellow) { /* implementation here */ }
    template<typename U> A& operator=(const A<U>& fellow) { /* implementation here */ }

    template<typename U> A(A<U>&& fellow) { /* implementation here */ }
    template<typename U> A& operator=(A<U>&& fellow) { /* implementation here */ }        
};

但是,出现以下错误

  

试图引用已删除的功能

当尝试将A个项目推到矢量或只是简单地复制构造时,例如:

A<int> a1{};
A<int> a2(a1);

UPDATE1:我需要模板copy&move构造函数和赋值运算符,因为template参数实际上仅控制某些缓存,因此A<T1>可以安全地分配给A<T2>

2 个答案:

答案 0 :(得分:5)

您可以通过声明具有备用签名的已删除副本构造函数/赋值运算符来使编译器满意,这不会导致选择此重载,但会阻止编译器生成构造函数/赋值运算符:

template<typename T> class A
{ public:
    A() { /* implementation here */ }

    // Remove from the implicit declaration of the default copy&move constructors and assignment operators
    A(A volatile const &) = delete;
    A & operator =(A volatile const &) = delete;

    // I want these to be used e.g. by std::vector
    template<typename U> A(A<U> const & fellow) { /* implementation here */ }
    template<typename U> A & operator =(A<U> const & fellow) { /* implementation here */ return *this;}

    template<typename U> A(A<U> && fellow) { /* implementation here */ }
    template<typename U> A & operator =(A<U> && fellow) { /* implementation here */ return *this; }        
};
int main()
{
    A<int> a1{};
    A<int> a2{a1};
    return 0;
}

online compiler

  

15.8.1复制/移动构造函数[class.copy.ctor]
  1.如果class X的非模板构造函数的第一个参数的类型为X&const X&volatile X&const volatile X&,则为复制构造函数没有其他参数,否则所有其他参数都有默认参数

答案 1 :(得分:1)

复制构造函数的最小示例,它使用第二个未使用(且默认)的参数将执行委托给模板构造函数

#include <iostream>

template <typename T>
struct A
 {
   A()
    { }

   A (A const & a0) : A{a0, 0}
    { }

   template<typename U>
   A (A<U> const &, int = 0)
    { std::cout << "template constructor" << std::endl; }
 };

int main()
 {
   A<int>  a0;
   A<int>  a1{a0};
 }

-编辑-

OP询问

  

operator=呢?尝试添加虚拟参数会导致编译器错误binary 'operator =' has too many parameters and 'operator =' cannot have default parameters

对于operator=(),我建议将两个运算符“委托”(在这种情况下不是委派构造函数的意思);一个模板。

类似

   template <typename U>
   A & assign (A<U> const &)
    { /* do assignment */ return *this; }

   A & operator= (A const & a0)
    { return assign(a0); }

   template <typename U>
   A & operator= (A<U> const & a0)
    { return assign(a0); }  

也许assign()方法可以是private方法。

或者更好,正如Jarod42所建议的(感谢),直接从非模板操作符调用模板操作符

template <typename U>
A & operator= (A<U> const & a0)
 { /* do assignment */ return *this; }

A & operator= (A const & a0)
 { return operator=<T>(a0); }