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

How to utilize template copy&move constructor and assignment operator?

提问人:Serge Rogatch 提问时间:1/5/2019 最后编辑:Serge Rogatch 更新时间:1/6/2019 访问量:204

问:

考虑以下 C++ 代码,我尝试避免非模板复制和移动构造函数和赋值运算符的首选失败:

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:我需要模板复制和移动构造函数和赋值运算符,因为模板参数实际上只是控制一些缓存,因此可以安全地分配给 .A<T1>A<T2>

C++ 模板 复制构造函数 赋值运算符 default-constructor

评论

0赞 Nelfeal 1/5/2019
复制/移动构造函数从来都不是模板。如果您不希望复制/移动构造函数完成这项工作,也许您可以委托。显示实施的哪一部分困扰着您;它可能最好放在任何构造函数之外。
1赞 user7860670 1/5/2019
A<int> a1();是一个函数声明(“最令人烦恼的解析...”),而且所有构造函数都是私有的
0赞 Jarod42 1/5/2019
而不是常规的复制/移动构造函数/赋值,将它们转发给模板实现。delete
0赞 Serge Rogatch 1/5/2019
@Jarod42,转发的具体含义是什么?似乎这就是我所要求的 - 如何转发。
1赞 max66 1/5/2019
@SergeRogatch - 您可以转发在模板构造函数中添加未使用的参数(带默认值)

答:

5赞 user7860670 1/5/2019 #1

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

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;
}

在线编译器

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

1赞 max66 1/5/2019 #2

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

#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); }

评论

0赞 Serge Rogatch 1/5/2019
怎么样?尝试添加虚拟参数会导致编译器错误和operator=binary 'operator =' has too many parameters'operator =' cannot have default parameters
0赞 max66 1/5/2019
@SergeRogatch - 答案有所改善;希望这会有所帮助。
0赞 Jarod42 1/6/2019
@max66:或者。return operator=<T>(rhs);
0赞 max66 1/6/2019
@Jarod42 - 你是对的:有效,而且更简单。谢谢。