使用“QWidget *QLayoutItem::widget()”会引发段错误(进入 qobjectdefs_impl.h)
Posted
技术标签:
【中文标题】使用“QWidget *QLayoutItem::widget()”会引发段错误(进入 qobjectdefs_impl.h)【英文标题】:Using "QWidget *QLayoutItem::widget()" throws a segment-fault (into qobjectdefs_impl.h) 【发布时间】:2019-02-08 15:46:35 【问题描述】:我在 QTabWidget 选项卡中使用 QTabWidget(s),当我尝试获取存储在“父”QTabWidget 选项卡布局中的“子”QTabWidget 时,出现段错误错误。
PS:我正在使用 Qt Creator v4.8.1、Qtv5.6.3 和 Windows 7 64 位操作系统(VBox v6.0.4 下的 VM)
最初,下面一行会抛出错误
dbTablesTabWidget = static_cast<QTabWidget*>(ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget());
我试过没有“static_cast”,但错误总是在这里
QWidget *widget = ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget();
我最后一次尝试(不返回 QWidget 指针)总是出现相同的“段错误”错误。
ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget();
我不明白的是以下行(第一个 for 循环中的第一行)
QString itemClass = ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget()->metaObject()->className();
虽然包含相同的“基本代码”,但效果很好。
段错误出现在“qobjectdefs_impl.h”文件中:
1 MainWindow::addRow mainwindow.cpp 298 0x403de6
2 QtPrivate::FunctorCall<QtPrivate::IndexesList<>, QtPrivate::List<>, void, void (MainWindow:: *)()>::call(void (MainWindow:: *)(), MainWindow *, void * *) qobjectdefs_impl.h 501 0x410175
3 QtPrivate::FunctionPointer<void (MainWindow:: *)()>::call<QtPrivate::List<>, void>(void (MainWindow:: *)(), MainWindow *, void * *) qobjectdefs_impl.h 520 0x4103e2
4 QtPrivate::QSlotObject<void (MainWindow:: *)(), QtPrivate::List<>, void>::impl(int, QtPrivate::QSlotObjectBase *, QObject *, void * *, bool *) qobject_impl.h 143 0x410243
5 QtPrivate::QSlotObjectBase::call qobject_impl.h 124 0x6ba5d241
6 QMetaObject::activate qobject.cpp 3715 0x6b8fc213
7 QMetaObject::activate qobject.cpp 3595 0x6b8fbc3e
8 QAbstractButton::clicked moc_qabstractbutton.cpp 307 0x9c0685
9 QAbstractButtonPrivate::emitClicked qabstractbutton.cpp 404 0x9be459
10 QAbstractButtonPrivate::click qabstractbutton.cpp 397 0x9be3f4
11 QAbstractButton::mouseReleaseEvent qabstractbutton.cpp 1002 0x9bf6f7
12 QWidget::event qwidget.cpp 8757 0x8f6881
13 QAbstractButton::event qabstractbutton.cpp 959 0x9bf56c
14 QPushButton::event qpushbutton.cpp 673 0xa55168
15 QApplicationPrivate::notify_helper qapplication.cpp 3804 0x8bf880
16 QApplication::notify qapplication.cpp 3277 0x8bd5f1
17 QCoreApplication::notifyInternal2 qcoreapplication.cpp 1015 0x6b8d37dd
18 QCoreApplication::sendSpontaneousEvent qcoreapplication.h 228 0xc2fb5d
19 QApplicationPrivate::sendMouseEvent qapplication.cpp 2773 0x8bc2cb
20 QWidgetWindow::handleMouseEvent qwidgetwindow.cpp 607 0x90f314
... <plus>
下面是 addRow 方法的摘录(单击 QPushButton 后调用的“自定义”私有插槽)
// var
int dbIndex = ui->tabWidget->currentIndex();
int tableIndex = 0;
QTabWidget *dbTablesTabWidget;
// Détection de l'index de l'item du layout de l'onglet de la BDD actuellement sélectionné
// -> Détection de la position du widget (jeu d'onglets des tables) dans le layout pour récupérer l'index de la table actuelle.
for (int i=0; i<ui->tabWidget->currentWidget()->layout()->count(); i++)
QString itemClass = ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget()->metaObject()->className();
qDebug(itemClass.toLatin1());
if (itemClass == "QTabWidget")
qDebug("Cet élément est un QTabWidget !!!");
//*** The following line throws a segment fault ***
//dbTablesTabWidget = static_cast<QTabWidget*>(ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget());
//QWidget *widget = ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget();
ui->tabWidget->currentWidget()->layout()->takeAt(i)->widget();
tableIndex = dbTablesTabWidget->currentIndex();
我是 Qt 的新手,我不知道段错误错误可能来自哪里。 编译过程OK,请看下图(法文版)
15:53:12: Configuration inchangée, étape qmake sautée.
15:53:12: Débute : "C:\Qt\Tools\mingw492_32\bin\mingw32-make.exe" -j1
C:/Qt/Tools/mingw492_32/bin/mingw32-make -f Makefile.Debug
mingw32-make[1]: Entering directory 'E:/Projects_Info_Slink/_Training/C++/QTCreatorProject/DataBaseViewerGUI/build-DataBaseViewerGUI-Desktop_Qt_5_6_3_MinGW_32bit-Debug'
mingw32-make[1]: Nothing to be done for 'first'.
mingw32-make[1]: Leaving directory 'E:/Projects_Info_Slink/_Training/C++/QTCreatorProject/DataBaseViewerGUI/build-DataBaseViewerGUI-Desktop_Qt_5_6_3_MinGW_32bit-Debug'
15:53:13: Le processus "C:\Qt\Tools\mingw492_32\bin\mingw32-make.exe" s'est terminé normalement.
15:53:13: Temps écoulé : 00:01.
【问题讨论】:
【参考方案1】:您在 tabwidget 中的页面可以包含任何小部件以及子选项卡小部件。
为了简单起见,或许你可以试试下面的方法:
//GET YOUR CURRENT PAGE IN FIRST TAB WIDGET
QWidget *widget = ui->tabWidget->currentWidget();
//FROM THE CURRENT PAGE FIND THE CHILD TAB WIDGETS
QList<QTabWidget*> tabWidgets = widget->findChildren<QTabWidget*>();
//USE THE FOUND WIDGET FOR YOUR PURPOSE. DO SOME SAFETY CHECKS FOR LIST.
QTabWidget* requiredTabWidget = tabWidgets[0];
【讨论】:
我在每个选项卡内都使用了一个布局,所以我无法直接获取任何 QWidget 子项(改为 QLayout 项)。如果不使用布局,我认为您的解决方案可能是一种解决方法! 我认为即使有布局,它也应该可以工作。请您试一次。【参考方案2】:我终于找到了错误:
我应该使用layout()->itemAt(i)
而不是layout()->takeAt(i)
。
它解释了为什么第二次发生错误。
takeAt
从列表中删除该项目,因此在下一次调用期间,该项目不存在于列表中。
PS : 对于 QList 对象,这是 at
而不是 itemAt
从元素列表中获取元素。
【讨论】:
以上是关于使用“QWidget *QLayoutItem::widget()”会引发段错误(进入 qobjectdefs_impl.h)的主要内容,如果未能解决你的问题,请参考以下文章
在使用加载数据流步骤的猪中,使用(使用 PigStorage)和不使用它有啥区别?
Qt静态编译时使用OpenSSL有三种方式(不使用,动态使用,静态使用,默认是动态使用)