欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

正确理解Widget::Widget(QWidget *parent) :QWidget(parent)这句话

程序员文章站 2022-06-24 11:11:51
原文:https://zhuanlan.zhihu.com/p/31310536 /********原文********/ 最近很多学习Qt的小伙伴在我的微信公众号私信我,该如何理解下面段代码的第二行QWidget(parent) 1 Widget::Widget(QWidget *parent) ......

原文:

/********原文********/

最近很多学习qt的小伙伴在我的微信公众号私信我,该如何理解下面段代码的第二行qwidget(parent)

1 widget::widget(qwidget *parent) :
2     qwidget(parent)
3 {
4 
5 }

 

为了统一回复大家,小豆君特意写了这篇文章,方便初学者们学习。

在讲解原因之前,先请大家看下面的一个例子

 1 #include <iostream>
 2 using namespace std;
 3 class base
 4 {
 5 public:
 6     base() :m_num(0){
 7         cout << "this is base()" << endl;
 8     }
 9     base(int val):m_num(val){
10         cout << "this is base(int val)" << endl;
11     }
12 private:
13     int m_num;
14 };

 

1 上方代码定义了一个基类base,并且有两个构造函数,一个是默认构造函数,一个是有一个整型参数的构造函数。

 

 1 class basechild: public base
 2 {
 3 public:
 4     basechild(){
 5         cout << "this is basechild()" << endl;
 6     }
 7     basechild(int val): base(val){
 8         cout << "this is basechild(val)" << endl;
 9     }
10 private:
11     int m_num;
12 };

 

2 上方代码定义了一个basechild类,并继承base类,同样的,它也定义了两个构造函数,一个默认,一个有整型参数。

 

1 int main(int argc, char *argv[])
2 {
3     basechild child1;
4     basechild child2(5);
5 
6     return 0;
7 }

3 main函数实例化了两个子类实例,child1,child2。child1调用默认构造函数。child2调用有整型参数的构造函数。

现在,我们运行程序,会有如下打印:

正确理解Widget::Widget(QWidget *parent) :QWidget(parent)这句话

看到了吗,我们发现:

  • 创建child1时,是先调用了base的默认构造函数,再调用自己的默认构造函数
  • 创建child2时,是先调用了base(int)这个构造函数,再调用自己的整型参数构造函数。

 

所以我们回头看basechild的构造函数

1 basechild(int val): base(val){
2         cout << "this is basechild(val)" << endl;
3     }

 

细心的同学,可能早就发现了,初始化列表中的base(val)正是调用了我们base基类的有参构造函数,而这样的写法就刚好是我们开头代码中的那段

1 widget::widget(qwidget *parent) :qwidget(parent)

 

所以widget是调用了qwidget下面的构造函数

1 qwidget(qwidget* parent = q_nullptr, qt::windowflags f = qt::windowflags());

 

所以得出如下总结:

总结: · 如果不指定构造函数,则派生类会调用基类的默认构造函数 · 派生类构造函数的初始化列表只能初始化派生类成员,不能直接初始化继承成员,如果想 要调用基类的有参构造函数,则可以在派生类的初始化列表中显示指定

以上总结,也告诉我们,当定义一个类时,最好为该类定义默认构造函数。

 

至此,我们明白了这个写法为什么会这样写。

 

好的,那么我们又提出一个问题,“调用qwidget(parent)这个构造函数,qwidget父类都做了哪些动作呢?”

下面是qwidget源码中的一部分节选:

 1 qwidget::qwidget( qwidget *parent, const char *name, wflags f )
 2     : qobject( parent, name ), qpaintdevice( pdt_widget ),
 3       pal( parent ? parent->palette()        // use parent's palette
 4            : *qapp->palette() )            // use application palette
 5 {
 6     if ( parent ) {
 7     qchildevent *e = new qchildevent( event_childinserted, this );
 8     qapplication::postevent( parent, e );
 9     }
10 }

 

大家从上面可以看出,如果parent参数非空的话,那么该构造函数使用了其父窗口的调色板,并且发送了qchildevent事件,这会让新的窗口成为parent所指窗口的子窗口,那么当父窗口被删除时,子窗口也会自动的被删除。

这其实是用到了qt对象树的概念,关于对象树,小豆君会在后面的分享中为大家介绍。