【问题标题】:C++11 vector of smart pointer智能指针的 C++11 向量
【发布时间】:2015-02-26 02:20:57
【问题描述】:

假设我们有以下代码。我们有以下课程

  • 抽象类的动物
  • Dog and Bird 是 Animal 的子类
  • 饲养所有动物的动物园

_

class Animal
{
public:
    Animal();
    void HasWings() = 0;
};

class Bird : public Animal
{
public:
    Bird() : Animal() {}
    void HasWings() override { return true; }
};

class Dog : public Animal
{
public:
    Dog() : Animal() {}
    void HasWings() override { return false; }
};

class Zoo
{
public:
    Zoo() {}
    void AddAnimal(Animal* animal) { _animals.push_back(animal); }
    ...
    std::vector<Animal*> _animals;
};

void myTest()
{
    Zoo myZoo;
    Bird* bird = new Bird();
    Dog* dog = new Dog();

    myZoo.AddAnimal(bird);
    myZoo.AddAnimal(dog);

    for (auto animal : myZoo._animals)
    {
        ...
    }
    ...
}

我希望用智能指针向量替换指针向量。即,

std::vector<std::shared_ptr<Animal>> _animals;

我们如何更改 Zoo 和 myTest 的代码? 我发现更新代码有困难,尤其是 Zoo 类中的“AddAnimal”方法

auto bird = std::make_shared<Bird>();
auto dog = std::make_shared<Dog>();
myZoo.AddAnimal(bird);
myZoo.AddAnimal(dog);

鸟和狗是不同的类型

【问题讨论】:

  • AddAnimal(std::shared_ptr&lt;Animal&gt; animal) - shared_ptr&lt;Bird&gt;shared_ptr&lt;Dog&gt; 将转换为 shared_ptr&lt;Animal&gt;,因为前两者是从后者派生的。 Animal::HasWings 也缺少 virtual 并且由于您已经有一个虚拟成员函数,您可能还应该使 Animal 的析构函数 virtual

标签: c++ c++11 abstract-class smart-pointers


【解决方案1】:

std::shared_ptr 的行为与 *-&gt; 运算符的原始指针的行为非常相似(实际上,取消引用运算符被“转发”到由 @ 存储的内部原始指针987654324@)。特别是,您可以将std::shared_ptr 用于基类,以便沿类层次结构进行虚拟调度。例如,下面的代码完全符合人们的假设,即在运行时调用适当的函数:

#include <iostream>
#include <memory>
#include <vector>

struct Base
{
    virtual void f() { std::cout << "Base::f()" << std::endl;}
    virtual ~Base() = default; // to silence -Wall warnings
};

struct Derived: Base
{
    void f() override { std::cout << "Derived::f()" << std::endl;}
};

int main()
{
    std::vector<std::shared_ptr<Base>> vsp; // access Derived via shared_ptr to Base

    auto base = std::make_shared<Base>();
    auto derived = std::make_shared<Derived>();

    vsp.push_back(base);
    vsp.push_back(derived);

    for(auto&& elem: vsp)
        elem->f(); // virtual dispatch
}

所以,大多数情况下,将Animal* 替换为std::shared_ptr&lt;Animal&gt; 就足够了,代码就可以正常工作了。 std::unique_ptr 有点复杂,因为后者是只能移动的类型(不能复制),所以一定要多加小心。

【讨论】:

  • 你是对的。如果我使用 std::unique_ptr 而不是 std::shared_ptr 因为我无法像过去一样创建返回 Animal 指针的方法。即“Animal* GetAnimal(int index);”
  • 说 Base 和 Derived 是小复杂的类,有一些函数和成员,你将如何,你将如何在 main 结束之前破坏向量 vsp 并避免内存泄漏?
  • @Annie 先做vsp.resize(0),然后你也可以做base.reset(); derived.reset(); 来释放初始共享指针(记住推入向量时引用计数加1)。
猜你喜欢
  • 2021-12-26
  • 1970-01-01
  • 2016-09-17
  • 2014-11-22
  • 2012-01-10
  • 1970-01-01
  • 2014-01-24
  • 2021-01-03
  • 1970-01-01
相关资源
最近更新 更多