5

我在 Qt 中构建了一个包含两个按钮的应用程序:一个退出按钮和一个导入按钮。按下导入按钮时,屏幕上的滚动区域中会显示按钮列表(文件 loggers.csv 包含数据 1;2;3;4;5;)。

一切正常,但是当我按下退出按钮(当然应该关闭所有内容)时,应用程序没有正确停止(Qt 的停止按钮仍然处于活动状态,而播放按钮没有)。当我运行调试器并按下退出按钮时,它会给出一个错误:指定给 RtlFreeHeap(0ADF0000, 0028FE40) 的地址无效。有谁能够帮助我?

主要的

#include <QtGui/QApplication>
#include "mainwindow.h"

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    MainWindow w;

    w.showFullScreen();

    return a.exec();
}

主窗口.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QtGui>
#include "logger.h"

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();

    QPushButton exit_btn;
    QPushButton import_btn;

private slots:

    void createMenus();
    void exit();
    void import();

private:

    int window_width;
    int window_height;

    int numLoggers;
    int numSelected;

    QVector<Logger*> loggers;

    QScrollArea * scroll_area;

    QVBoxLayout scrollLayout;

    QWidget viewport;

    Ui::MainWindow *ui;
};

#endif // MAINWINDOW_H

主窗口.cpp:

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include "QtGui"

MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    window_width = QApplication::desktop()->width();
    window_height = QApplication::desktop()->height();

    createMenus();

    connect(&exit_btn,SIGNAL(clicked()),this,SLOT(exit()));
    connect(&import_btn,SIGNAL(clicked()),this,SLOT(import()));
}

MainWindow::~MainWindow()
{
   delete ui;
}

void MainWindow::createMenus()
{
    import_btn.setParent(ui->centralWidget);
    import_btn.setGeometry(400,300,100,100);
    import_btn.setText("IMPORT");

    exit_btn.setText("EXIT");
    exit_btn.setParent(ui->centralWidget);
    exit_btn.setGeometry(window_width-50,12,32,32);

    viewport.setLayout(&scrollLayout);
    viewport.resize(0,0);

    scroll_area = new QScrollArea(ui->centralWidget);
    scroll_area->setGeometry(0,66,317,window_height-116);
    scroll_area->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
    scroll_area->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
    scroll_area->setWidget(&viewport);
    scroll_area->setGeometry(0,97,317,window_height-228);

    scrollLayout.setMargin(0);
    scrollLayout.setSpacing(0);
}

void MainWindow::exit()
{
    close();
    qApp->quit();
}

void MainWindow::import()
{
    numSelected=0;

    QFile f("Loggers3.csv");

    if (f.open(QIODevice::ReadOnly))
    {
        numLoggers=0;

        QString data;
        data = f.readAll();
        QStringList vals = data.split(';');

        while(vals.size()>=1)
        {
            Logger * logger = new Logger;

            logger->setNumber(vals[0].toInt());
            vals.removeAt(0);

            loggers<<logger;

            numLoggers++;
        }
        f.close();


        for(int i=0; i<numLoggers;i++)
        {
            loggers[i]->createButtons();
            scrollLayout.addWidget(loggers[i]->button);
        }

        viewport.resize(367,numLoggers*60);
    }
}

记录器.h

#ifndef LOGGER_H
#define LOGGER_H

#include <QtGui>

class Logger : public QWidget
{
    Q_OBJECT
public:
    explicit Logger(QWidget *parent = 0);

    ~Logger();
    int number;
    QLabel num;
    QToolButton * button;
    bool checked;

signals:

public slots:

    void setNumber(int number);
    void createButtons();
};

#endif // LOGGER_H

记录器.cpp

#include "logger.h"
#include <QtGui>

Logger::Logger(QWidget *parent) :
    QWidget(parent)
{
    button = new QToolButton;
    button->setCheckable(true);
    button->setMinimumSize(317,60);
    button->setStyleSheet("QToolButton{background-image: url(images/btn_bg); border:none}");
}

Logger::~Logger()
{
}

void Logger::setNumber(int logNumber)
{
    number=logNumber;
}

void Logger::createButtons()
{
    QLayout * layout = new QHBoxLayout;

    QSpacerItem *spacer = new QSpacerItem(120, 31, QSizePolicy::Maximum, SizePolicy::Maximum);

    num.setStyleSheet("color: white; font: bold 16px");
    num.setText(QString::number(number));

    layout->addWidget(&num);
    layout->addItem(spacer);

    button->setLayout(layout);
}
4

3 回答 3

6

我不完全确定您要达到的目标...但是您的问题在于这两行:

viewport.setLayout(&scrollLayout);
viewport.resize(0,0);

在 QWidget 类的文档中,它指出:

如果这个小部件上已经安装了布局管理器,QWidget 不会让你安装另一个。您必须先删除现有的布局管理器(由 layout() 返回),然后才能使用新布局调用 setLayout()。

这就是你的问题所在。不信,在这两行代码之前加上这个检查。

 if(layout()){
        qDebug() << "Another layout exists";
    }

来源:QVBoxLayout 类参考

QVBoxLayout 类垂直排列小部件。

此类用于构造垂直框布局对象。有关详细信息,请参阅 QBoxLayout。

该类的最简单用法是这样的:

 QWidget *window = new QWidget;
 QPushButton *button1 = new QPushButton("One");
 QPushButton *button2 = new QPushButton("Two");
 QPushButton *button3 = new QPushButton("Three");
 QPushButton *button4 = new QPushButton("Four");
 QPushButton *button5 = new QPushButton("Five");

 QVBoxLayout *layout = new QVBoxLayout;
 layout->addWidget(button1);
 layout->addWidget(button2);
 layout->addWidget(button3);
 layout->addWidget(button4);
 layout->addWidget(button5);

 window->setLayout(layout);
 window->show();

首先,我们在布局中创建我们想要的小部件。然后,我们创建 QVBoxLayout 对象并将小部件添加到布局中。最后,我们调用 QWidget::setLayout() 将 QVBoxLayout 对象安装到小部件上。此时,布局中的小部件被重新设置为将窗口作为其父级。


项目中的关键错误来源:

小部件应该在堆上构建,因为当它们的父级被删除时,它们会被自动删除。 您有一个在堆上实例化的自定义小部件类。成员也应该放在堆上。此外,您应该考虑在 GUI 代码中使用父/子层次结构,以确保正确的内存管理和正确的删除。

于 2012-10-01T18:40:01.820 回答
1

以我的经验,如果您的程序停止,RtlFreeHeap这是内存损坏的好兆头。

打电话时

import_btn.setParent(ui->centralWidget);

centralWidget拥有import_btn. 这意味着,当centralWidget被删除时(作为delete ui;MainWindow的析构函数的一部分发生),它将调用delete您的成员变量!

这会导致报告的内存损坏。

您需要QPushButton动态分配 's,而不是作为普通成员变量。所以让他们QPushButton*

于 2012-10-02T10:24:54.600 回答
0

以下是我从 mainwindow.cpp 执行此操作的方法,这要感谢这个问题:How to create a correct exit button in qt

QPushButton * quit_btn = new QPushButton(this);
quit_btn->setGeometry(540,440,93,27);
quit_btn->setText("Exit");
QObject::connect(quit_btn,SIGNAL(clicked()),qApp,SLOT(quit()));

完美运行:D

于 2014-03-14T09:35:32.107 回答