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

 

该如何理解下面段代码的第二行QWidget(parent)


 
 
  1. 1 Widget::Widget(QWidget * parent) :
  2. 2 QWidget( parent)
  3. 3 {
  4. 4 }

 

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


 
 
  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. return 0;
  6. }

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

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

看到了吗,我们发现:

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

 

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


 
 
  1. BaseChild( int val): Base(val){
  2. cout << "this is BaseChild(val)" << endl;
  3. }

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

Widget::Widget(QWidget *parent) :QWidget(parent)

 
 

所以Widget是调用了QWidget下面的构造函数

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所指窗口的子窗口,那么当父窗口被删除时,子窗口也会自动的被删除。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值