Qt:子对象可以在其父对象中组合吗?
在 Qt 中,我可以通过组合将子窗口小部件嵌入到其父窗口小部件中,还是必须使用 new
创建它们?
class MyWindow : public QMainWindow
{
...
private:
QPushButton myButton;
}
MyWindow::MyWindow ()
: mybutton("Do Something", this)
{
...
}
文档说,任何从QObject
派生的对象都会在其父对象被销毁时自动销毁;这意味着对 delete
的调用,在上面的示例中会崩溃。
我必须使用以下内容吗?
QPushButton* myButton;
myButton = new QPushButton("Do Something", this);
编辑
答案多种多样,基本上可以归结为三种可能性:
- 是,构图还可以。 Qt 可以弄清楚对象是如何分配的,并且只能
删除
堆分配的对象(这是如何工作的?) - 是,组合可以,但不指定父级,因为父级会在对象上调用
delete
(但是无父级窗口小部件不会变成顶级窗口吗?) - 不,窗口小部件始终必须是堆分配的。
哪一个是正确的?
In Qt, can I embed child widgets in their parent via composition, or do I have to create them with new
?
class MyWindow : public QMainWindow
{
...
private:
QPushButton myButton;
}
MyWindow::MyWindow ()
: mybutton("Do Something", this)
{
...
}
The documentation says that any object derived from QObject
will automatically destroyed when its parent is destroyed; this implies a call to delete
, whcih in the above example would crash.
Do I have to use the following?
QPushButton* myButton;
myButton = new QPushButton("Do Something", this);
EDIT
The answers are quite diverse, and basically boil down to three possibilities:
- Yes, composition is ok. Qt can figure out how the object was allocated and only
delete
heap-allocated objects (How does this work?) - Yes, composition is ok, but don't specify a parent, since the parent would otherwise call
delete
on the object (But won't a parent-less widget turn into a top-level window?) - No, widgets always have to be heap-allocated.
Which one is correct?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(7)
当该特定对象的删除序列开始时,非静态、非堆成员变量将被删除。只有当所有成员都被删除后,才会进入基类的析构函数。因此,QPushButton myButton 成员将在调用 ~QMainWindow() 之前被删除。从 QObject 文档中可以看出:“如果我们在父对象之前删除子对象,Qt 会自动从父对象的子对象列表中删除该对象”。因此不会发生崩溃。
The non-static, non-heap member variables are deleted when that particular object's delete sequence starts. Only when all members are deleted, will it go to the destructor of the base class. Hence QPushButton myButton member will be deleted before ~QMainWindow() is called. And from QObject documentation: "If we delete a child object before its parent, Qt will automatically remove that object from the parent's list of children". Hence no crash will occur.
对象树和对象树所有权回答了你的问题。基本上,当子对象在堆上创建时,它将被其父对象删除。
另一方面,当在堆栈上创建子对象时,销毁顺序很重要。子级将在其父级之前被销毁,并将其自身从其父级列表中删除,以便其析构函数不会被调用两次。
该链接中还有一个示例,显示了有问题的销毁顺序。
Object trees & ownership answers your question. Basically when the child object is created on the heap it will be deleted by its parent.
On the other hand when the child object is created on the stack the order of destruction is important. The child will be destroyed before its parent and will remove itself from its parent's list so that its destructor is not called twice.
There is also an example in that link that shows problematic order of destruction.
否。它意味着调用该特定实体的析构函数。
在您的示例中,如果
MyWindow
被销毁,则意味着MyWindow
的析构函数已被调用。反过来,它会调用已经在QPushButton
中实现的析构函数myButton
。如果您有复合实体,则只会在该实体上调用析构函数,而不是
删除
,因此它不会崩溃。Qt 中的父子关系不需要专门位于堆栈或堆中。它可以存在于任何事物中。
堆栈上父子关系的类似示例是 这里。
哈..
No. It implies a call to the destructor of that particular entity.
Say in your example, if
MyWindow
is destroyed, it means the destructor of theMyWindow
has been called. Which in turn will call the destructormyButton
which is implemented already inQPushButton
.If you have composite entity, just the destructor will be called on that entity but not
delete
and so it won't crash.Parent child relationships in Qt doesn't require specifically to be in a stack or heap. It can be in anything.
A similar example in parent child relationship over a stack is over here.
HTH..
该对象只有当它有父指针时才会被销毁,所以你可以使用:
The object will be destroyed only when it has a parent pointer, so you can use:
您应该在堆上创建它,因为 QObject 会销毁它:
You should create it on heap, since QObject will destroy it :
调用
delete
运算符不会使您的应用程序崩溃,您可以阅读以下引用请注意,父参数默认为 NULL(默认参数)
这是 QPushButton 构造函数,
因此您可以使用
并且可以在任何组件上调用
delete
,并且Qt 会随时处理这一点
calling
delete
operator will not crash ur application, you can read the following quotenote that the parent argument is
NULL
by default (default argument)this is QPushButton Constructor
so u can use
and u can call
delete
on any component and any timeQt will take care for this point
让我在这里引用来源 。
正如您所看到的,
QObject
确实在析构函数中删除了
它的每个子级。此外,析构函数执行之前 任何成员的析构函数;因此,如果所讨论的组合等于父级 - 那么成员QObject
将没有任何机会将其自身从其父级的子级列表中删除。不幸的是,这意味着您无法将
QObject
组合到其父级中。但是您可以组合成其他对象,也可以在堆栈上分配 - 只要您保证在父对象开始破坏之前将其父对象重置为 0 即可。Let me just quote the source here.
So as you can see,
QObject
does indeeddelete
each of its children in the destructor. In addition, the destructor is executed before destructors of any members; so if the composite in question equals the parent – then memberQObject
won't have any chance to remove itself from the children list of its parent.This, unfortunately, means that you cannot compose a
QObject
into its parent. But you can compose into other objects, as well as allocate on stack – as soon as you guarantee to destruct the object or reset its parent to 0 before the parent starts destructing.