QSplitter 控件使用时要注意的问题

当使用QSplitter时,如果是多个QSPlitter嵌套使用(如下图)


就很容易出现错误。通常是“内存不能为read” (如下图)

代码如下:(错误的用法)

[cpp]  view plain copy
  1. #include <QApplication>  
  2. #include <QFont>  
  3. #include <QPushButton>  
  4. #include <QSplitter>  
  5. #include <QTextEdit>  
  6. #include <QTableWidget>  
  7. #include <QTreeWidget>  
  8. #include <QSettings>  
  9.   
  10. int main(int argc, char* argv[])  
  11. {  
  12.     QApplication app(argc, argv);  
  13.   
  14.     QTextEdit *editor1 = new QTextEdit;  
  15.     QTableWidget *table = new QTableWidget;  
  16.     QTreeWidget *tree = new QTreeWidget;  
  17.   
  18.     //子splitter  
  19.     QSplitter vSplitter(Qt::Vertical);  
  20.     vSplitter.addWidget(table);  
  21.     vSplitter.addWidget(editor1);  
  22.     //父splitter  
  23.     QSplitter hSplitter(Qt::Horizontal);  
  24.     hSplitter.addWidget(tree);  
  25.     hSplitter.addWidget(&vSplitter);  
  26.   
  27.     hSplitter.show();  
  28.   
  29.     return app.exec();  
  30. }  


出现错误的原因为,如果先创建的是父QSplitter,然后创建的子QSplitter,则没有错误。

因为当关闭窗体是,调用析构函数的顺是:子,父。

但是如果是先创建的子QSplitter,然后创建的父QSplitter,那么在关闭窗体时,

先调用的是父QSplitter的析构函数,然后才调用的子QSplitter的析构函数。

这样就会出错。个人猜测试由于在父QSplitter的析构函数中已经自动调用了子QSplitter的析构函数,

当再次调用子QSplitter的析构函数时,就会找不到对象,导致出错。

这就要求我们一定要按顺序定义QSplitter,这很难做到。

相反,如果我们自己决定调用析构函数的顺序就不会出错了。

因此,正确的使用QSplitter的方法是在堆上建立对象,然后自行delete。

正确用法如下:

[cpp]  view plain copy
  1. include <QApplication>  
  2. #include <QFont>  
  3. #include <QPushButton>  
  4. #include <QSplitter>  
  5. #include <QTextEdit>  
  6. #include <QTableWidget>  
  7. #include <QTreeWidget>  
  8. #include <QSettings>  
  9.   
  10. int main(int argc, char* argv[])  
  11. {  
  12.     QApplication app(argc, argv);  
  13.   
  14.     QTextEdit *editor1 = new QTextEdit;  
  15.     QTableWidget *table = new QTableWidget;  
  16.     QTreeWidget *tree = new QTreeWidget;  
  17.   
  18.     //子splitter  
  19.     QSplitter *vSplitter = new QSplitter(Qt::Vertical);  
  20.     vSplitter->addWidget(table);  
  21.     vSplitter->addWidget(editor1);  
  22.     //父splitter  
  23.     QSplitter *hSplitter = new QSplitter(Qt::Horizontal);  
  24.     hSplitter->addWidget(tree);  
  25.     hSplitter->addWidget(vSplitter);  
  26.   
  27.     hSplitter->show();  
  28.   
  29.     app.exec();  
  30.   
  31.     delete vSplitter;   //先删除子  
  32.     delete hSplitter;   //后删除父  
  33.   
  34.     return 0;  
  35. }  

ps:

由于调用父splitter会自动删除其内部的所有控件,所以也可以直接delete父QSplitter。

这样连里面的其他控件也会删除了。但是切记,一定要用new。

ps:

类似的布局类控件如QScrollArea也要注意相同的问题。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值