4

我正在尝试将一个小型实用程序应用程序更新为更现代的 C++ 方式,但是我在使用一些带有 std::shared_ptr 的 Qt 对象时遇到了问题,尤其是那些接收一些其他 QWidget 作为构造函数参数的对象。

例如:

private:
    std::shared_ptr<QWidget> centralwidget;
    std::shared_ptr<QVBoxLayout> verticalLayout;

public:
    void setupUi(QMainWindow *MainWindow) // this pointer is a .get() from a shared_ptr
    {
        centralwidget = std::make_shared<QWidget>(new QWidget(MainWindow)); // compiles fine
        verticalLayout = std::make_shared<QVBoxLayout>(new QVBoxLayout(centralwidget.get())); // does not compile
    }

编译错误是:

错误 1 ​​错误 C2664: 'QVBoxLayout::QVBoxLayout(QWidget *)' : 无法将参数 1 从 'QVBoxLayout *' 转换为 'QWidget *' e:\microsoft visual studio 11.0\vc\include\memory 855

我似乎无法理解这个错误,我没有转换任何东西,我只是想创建一个 QVBoxLayout 对象并将 QWidget 作为其父级传递(就像我对原始指针所做的那样)。

4

2 回答 2

14

一般来说,我尽量避免对 Qt GUI 对象使用 shared_ptr,因为 Qt 已经提供了自己的内存管理机制。每个 QObject 都可能有一个父对象,当这个父对象死亡时,他会删除它的所有子对象。这里不需要 shared_pointer 并且不会给您任何附加值:您可以完美地使用原始指针而不会造成内存泄漏。

一般来说,如果 QObject 的父对象在最后一个 shared_ptr 实例被删除之前就死掉了,你很快就会遇到麻烦,因为当最后一个 shared_ptr 被销毁时,该对象将被第二次删除。这不是这里的情况,但要小心:)

于 2012-06-09T19:06:28.200 回答
5

的参数std::make_shared被传递给您正在实例化的类的构造函数。所以基本上你在做什么相当于:

new QVBoxLayout(new QVBoxLayout(centralwidget.get()))

我认为你想要做的是:

centralwidget = std::make_shared<QWidget>(MainWindow);
verticalLayout = std::make_shared<QVBoxLayout>(centralwidget.get());

查看std::make_shared(例如此处)的文档。这个函数的重点是在内存中的对象实例附近分配引用计数,所以你必须让它为你分配。

于 2012-06-09T18:08:39.140 回答