std::list 模板是否需要其实例类型中的复制构造函数(或等效项)?

Does std::list template require a copy constructor (or equivalent) in its instance type?

本文关键字:复制 构造函数 类型 是否 list std 实例      更新时间:2023-10-16

我有一个类,每个实例都要被考虑在内,创建和销毁受到严格监管。不允许随机移动、复制、临时 - 一旦通过专用函数创建,实例只能通过引用和指针"传递"。

为了实现这一点,我删除了这个类的复制构造函数和赋值运算符。

这些实例应该保存在 std::list 中,由 emplace_back() 创建,按需删除,并且从未打算移动。但是我收到有关已删除的复制构造函数的错误。

In constructor 'std::_List_node<_Tp>::_List_node(_Args&& ...) 
error: deleted function 'Reader::Reader(const Reader&) 
stl_list.h:103: error: used here

有没有办法做到这一点?std::list 的一些替代品,我不需要手工雕刻?

标题中问题的答案是"视情况而定"。

如果您的类确实有移动构造函数,您将能够使用 move 构造函数。如果您的类没有移动构造函数,则将使用复制构造函数。

必须以某种方式构造列表中的对象。emplace_back使其尽可能高效,但它仍然需要构造一个对象。从 http://en.cppreference.com/w/cpp/container/list/emplace_back:使用放置新在容器提供的位置就地构造元素。

当要emplace_back的参数是另一个对象时,放置 new 最终将调用复制构造函数或移动构造函数。

如果要emplace_back的参数只是构造对象所需的数据,则不需要复制构造函数或移动构造函数。


有没有办法做到这一点?

如果您或您的团队关于复制构造函数和移动构造函数的策略未公开讨论,则必须使用解决方法。

解决方法 1:

存储指向列表中对象的指针。这是最简单的解决方法,只要您可以确保对象不会在列表背后删除,并使列表保持悬空指针。

解决方法 2:

存储列表中对象的非指针句柄。这将需要您进行一些簿记代码。如果可以添加以下功能:

  1. 获取给定指向对象的指针的整数值。整数值可用作对象的非指针句柄。
  2. 获取指向给定先前返回句柄的对象的指针。

您可以使用句柄列表轻松管理。

通过保留几个随着对象的构造和删除而更新的地图,可以轻松完成上述工作。每次创建对象时,都会向地图添加条目。每次删除对象时,都会从地图中删除条目。