c++ - 在需要之前避免初始化成员

标签 c++ class memory c++11 initialization

我正在创建一个通用 AtomManager<T>可存放Atom<T>的容器对象。

enum State { Alive, Dead, Unused };
template<class T> struct Atom
{
    T impl;
    int index, counter;
    State state;
};

我要impl不要放在堆上,因为我将存储 Atom<T>实例在管理器中连续。

AtomManager<T>将原子存储在std::vector<Atom<T>>中像这样:

| A | A | A | A | A | A | U | U | U | U |

哪里A意味着活着,并且 U表示未使用。当用户调用AtomManager<T>::refresh()时,所有带有 state 的原子等于 State::Dead将被移动到存储的末尾,然后它们将被设置为 State::Unused 。示例:

| A | A | A | A | A | A | U | U | U | U |

// some atoms die

| A | D | A | D | A | A | U | U | U | U |

// user calls refresh()

| A | A | A | A | U | U | U | U | D | D |

// after refresh()

| A | A | A | A | U | U | U | U | U | U |

为了创建原子,我有一个与 T 匹配的函数借助可变参数模板的构造函数签名并构造 T在从存储开始处开始的第一个未使用的原子中。

问题是 T必须是默认可构造(因为我在 resize() 上调用 std::vector )。但这是我不需要的,因为我只关心 Atom<T>::impl当状态为 State::Alive 时或State::Dead 。如果原子是活的还是死的,则意味着用户之前使用前面提到的可变参数函数构造了它。

但是,我确实关心Atom<T>::indexAtom<T>::counter当原子未使用时(并且 Atom<T>::impl 是垃圾)。

我不关心 Atom<T>::impl 的状态当原子未使用时。

但是编译器会这样做。我无法使用AtomManager<T>T不可默认构造。

我尝试在Atom<T>内使用 union :

union { T impl; char dummy; };

...但我无法让它正常工作。

如何存储 T Atom<T> 内未构造的未初始化实例?

我不关心它的状态。我确信在访问它之前我会正确构建它。但是当原子不使用时,我希望它处于未定义状态。

实现这一目标的最佳方法是什么?我不想要 Atom<T>::impl存储在堆上。

我不想引入额外的依赖项。我不需要查询Atom<T>::impl的状态据我知道何时可以安全访问它。

最佳答案

简单的答案是使用 boost::Optional,但您说由于某种原因您不希望这样做。

要推出您自己的可选类型,您需要一个字节数组,并与 T 适当对齐。在 C++11 或更高版本中,这很简单:

alignas(T) char bytes[sizeof(T)];

如果您坚持使用历史方言,那么您可能必须使用特定于编译器的扩展来指定对齐方式,或者只是希望得到最好的结果。

现在您可以使用placement-new创建对象:

T * impl = new(bytes) T(...);

访问它的最简单方法是通过函数:

T & get_impl() {return *reinterpret_cast<T*>(bytes);}

并且不要忘记销毁它(但前提是您创建了它):

get_impl().~T();

关于c++ - 在需要之前避免初始化成员,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23518073/

相关文章:

c++ - 类返回值错误

c++ - 为什么函数范围内不允许类前向声明​​?

java - 想要在本地计算机的 JBoss 服务器中分析 Web 应用程序

c++ - 继承C++

c++ - VSCode : IDE does not highlight errors in C++ code anymore

javascript - 我们可以在 javascript 的构造函数中放入什么变量?

node.js - 限制 node.js 的内存使用

c++ - 通过 C++ 中的 SOCI/ODBC 的 SQL Server 存储过程

c++ - 刷新用作 QTableView 编辑器的 QComboBox 中的选项

memory - 为什么 Erlang 会在大序列上崩溃?