如何使用unique_ptr和make_unique正确声明对象数组



>我正在尝试声明和初始化一个保存类数组的unique_ptr

这是我用来解决项目内存管理问题的示例。我可以声明指针,但无法初始化它。

class CrewMember
{   
};

class SpaceShip
{
// generates error
std::unique_ptr<CrewMember[3][3]> ship_crew_members = std::make_unique< new CrewMember[3][3]>; 
// compiles fine
std::unique_ptr<CrewMember[3][3]> ship_crew_members;
};

收到的错误:

调用非 constexpr 函数 'void* 运算符新 ' std::unique_ptr ship_crew_members = std::make_unique<新船员[3][3]>;
^
无法根据转换为类型"std::unique_ptr"来解决重载函数"make_unique" std::unique_ptr ship_crew_members = std::make_unique<新船员[3][3]>;

  1. std::unique_ptr存储指向单个对象或一维数组的指针。不支持多维数组。理论上,如果你做一个邪恶的投射,它可以存储一个指向多维数组的指针,但你真的不应该。

  2. 如果你真的想用std::unique_ptr来存储一个多维数组,你需要做一些额外的工作,使用一维数组,然后自己计算多维索引的偏移量(例如[1][2]变成1*3+2)。

  3. std::make_unique是一个函数模板。模板参数需要是一个正确的类型,它可以是一个类或一个类数组(但没有大小)。如果您的类型是一个数组,并且您为std::make_unique指定一个大小作为参数,它将创建一个指向该大小的数组的std::shared_ptr,并在该数组被销毁时在该数组上使用正确的delete[]运算符。

总之,看起来像这样:

std::shared_ptr<CrewMember[]> ship_crew_members = std::make_shared<CrewMember[]>(9);

但是,我强烈建议不要这样做。更简单的是使用任何标准容器,例如std::vector

std::vector<std::vector<CrewMember>> ship_crew_members;

这为您提供了一个不同大小的多维容器,您可以在其中添加元素或从中删除元素。

或者std::array固定大小的多维数组:

std::array<std::array<CrewMember, 3>, 3> ship_crew_members;

相关内容

  • 没有找到相关文章

最新更新