Ruslan Polupan
Ruslan Polupan9. Januar 2018 04:45

Исключения. Потоки.

потоки, qt, exception, Linux

Делаю небольшую утилиту по работе с базой Fierbird.

Под MinGW на win компилировалось без проблем.
На работе попробовал собрать про Linux выдало следующее:
Warning: Qt has caught an exception thrown from an event handler. Throwing
exceptions from an event handler is not supported in Qt.
You must not let any exception whatsoever propagate through Qt code.
If that is not possible, in Qt 5 you must at least reimplement
QCoreApplication::notify() and catch all exceptions there.

Подозреваю что что-то с потоками. Ибо только начал с ними разбираться. В потоке создаю подключение к базе данных для дальнейшего использования.

Класс подключения который передаю в поток.

dbaseconnect.h

#ifndef DBASECONNECT_H
#define DBASECONNECT_H

#include <QObject>
#include <QMap>

class DBaseConnect : public QObject
{
    Q_OBJECT
public:
    explicit DBaseConnect(QMap<QString,QString> opt, QObject *parent = nullptr);
signals:
    void connectionError(QString errorMeaage);
    void fin();
    void sendStatus(bool);
public slots:
    void createConnection();
private:
    QMap<QString, QString> config;
};

#endif // DBASECONNECT_H
dbaseconnect.cpp
#include "dbaseconnect.h"
#include <QtSql>
#include <QDebug>

DBaseConnect::DBaseConnect(QMap<QString,QString> opt, QObject *parent) : QObject(parent)
{
    config = opt;
}

void DBaseConnect::createConnection()
{

    ///Подключаемся к базе данных АЗС
    QSqlDatabase db = QSqlDatabase::addDatabase("QIBASE","central");
    db.setHostName(config.value("server"));
    db.setDatabaseName(config.value("basename"));
    db.setUserName(config.value("login"));
    db.setPassword(config.value("password"));
    if(!db.open()) {
        emit sendStatus(false);
        QString errorString =  db.lastError().text();
        qDebug() <<  "Не возможно подключиться к базе данных." << endl << "Причина:" << errorString;

        emit connectionError(errorString);
        emit fin();
    }
    emit sendStatus(true);
    emit fin();
}

Подключение по кнопке в диалоге со писком подключений.
void ConnectionDialog::on_pushButtonConnect_clicked()
{
    thread = new QThread;
    progress = new QProgressDialog();
    isConnected =true;
    progress->setWindowModality(Qt::WindowModal);
    progress->setLabelText("Подключение к центральной базе данных...");
    progress->setCancelButton(0);
    progress->setRange(0,0);
    progress->setMinimumDuration(0);

    database.insert("server", ui->lineEditServer->text().trimmed());
    database.insert("basename",ui->lineEditDataBase->text().trimmed());
    database.insert("login",ui->lineEditLogin->text().trimmed());
    database.insert("password",ui->lineEditPassword->text().trimmed());

    DBaseConnect *dbConn = new DBaseConnect(database);

    connect(thread,SIGNAL(started()),this,SLOT(startDBConnect()));
    connect(thread,SIGNAL(started()),dbConn,SLOT(createConnection()));
    connect(thread,SIGNAL(finished()),this,SLOT(finishDBConnect()));
    connect(dbConn,SIGNAL(sendStatus(bool)),this,SLOT(getStaus(bool)));
    connect(dbConn,SIGNAL(connectionError(QString)),this,SLOT(errogConnectInfo(QString)));
    connect(dbConn,SIGNAL(fin()),thread,SLOT(terminate()));


    thread->start();
    dbConn->moveToThread(thread);
}
Слоты
void ConnectionDialog::startDBConnect()
{
    progress->show();
}

void ConnectionDialog::finishDBConnect()
{
    progress->cancel();
    if(isConnected) {
        this->accept();
    }
}

void ConnectionDialog::errogConnectInfo(QString str)
{
    QMessageBox::critical(0, qApp->tr("Не могу открыть базу данных"),
                              QString("Не могу установить соединение с центральной БД!\nПричина: %1\n Проверьте настройки подключения.").arg(str),
                              QMessageBox::Ok);
    isConnected=false;
}

void ConnectionDialog::getStaus(bool status)
{
    isConnected =status;
}

Подскажите что я делаю не так.

ЗЫ: Программирование это для меня небольшое хобби. Так что буду рад любым комментариям.
Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.

Magst du es? In sozialen Netzwerken teilen!

8
Evgenii Legotckoi
  • 9. Januar 2018 04:53
  • (bearbeitet)

Добрый день.
Вы можете с помощью дебаггера найти строку, которая выкидывает исключение в вашем коде?
Пройдитесь по стеку вызовов функций и посмотрите в каком именно месте вашего кода происходит исключение.
Драйвер FireBird под Linux нашли/собрали?

Сам по себе код выглядит корректным, скорее всего платформозависимые особенности при компиляции.

    Ruslan Polupan
    • 9. Januar 2018 06:31

    Драйвер собран и работает.

    Пройдитесь по стеку вызовов функций
    Можно немного поподробнее :-)

    Методом исключения, очень похоже, что проблемы при завершении работы потока.
      Evgenii Legotckoi
      • 9. Januar 2018 06:46

      Конечно, можно.
      Запустите программу в режиме отладки, в Qt Creator это горячая клавиша F5, дойдите в программе до момента падения, когда будет вылетать исключение. Когда вылетит исключение, автоматически вас перекинет в Qt Creator и там будет таблица со стеком вызовов всех функций. Внимательно посмотрите её, там будут как ваши функции и методы, так и Qt-шные, также вполне возможно, что куда-то в драйвер будет уходит стек вызовов. Возможно, это натолкнёт вас на мысль, что не так.


      По поводу потоков и исключений всегда трудно что-то сказать, особенно когда неизвестно, откуда прилетает исключение. Здесь в первую очередь дебажить нужно.
        Ruslan Polupan
        • 9. Januar 2018 07:27
        Про отладчик в общих чертах я в курсе. :-)
        Таблицу со стеком искал :-)
        Поставил в настройках Q Creater Отладка "Останавливаться на вызове qWarning".
        Warning появляется при вызове слота terminate.

          Evgenii Legotckoi
          • 9. Januar 2018 07:38

          Знаете, что мне в этом методе сейчас не понравилось?

          void DBaseConnect::createConnection()
          {
          
              ///Подключаемся к базе данных АЗС
              QSqlDatabase db = QSqlDatabase::addDatabase("QIBASE","central");
              db.setHostName(config.value("server"));
              db.setDatabaseName(config.value("basename"));
              db.setUserName(config.value("login"));
              db.setPassword(config.value("password"));
              if(!db.open()) {
                  emit sendStatus(false);
                  QString errorString =  db.lastError().text();
                  qDebug() <<  "Не возможно подключиться к базе данных." << endl << "Причина:" << errorString;
          
                  emit connectionError(errorString);
                  emit fin();
              }
              emit sendStatus(true);
              emit fin();
          }
          У вас сигнал emit fin(); вызывается дважды, если не удалось открыть соединение с базой данных. Это нехорошо, дважды пытаться вызвать слот terminate(), на второй раз возможно даже и вызывать нечего.
          Перепишите так, например, код после условия не должен выполняться
          void DBaseConnect::createConnection()
          {
          
              ///Подключаемся к базе данных АЗС
              QSqlDatabase db = QSqlDatabase::addDatabase("QIBASE","central");
              db.setHostName(config.value("server"));
              db.setDatabaseName(config.value("basename"));
              db.setUserName(config.value("login"));
              db.setPassword(config.value("password"));
              if(!db.open()) {
                  emit sendStatus(false);
                  QString errorString =  db.lastError().text();
                  qDebug() <<  "Не возможно подключиться к базе данных." << endl << "Причина:" << errorString;
          
                  emit connectionError(errorString);
                  emit fin();
                  return;
              }
              emit sendStatus(true);
              emit fin();
          }
          Либо перепешите метод как-то иначе так, чтобы fin() всегда вызывался один раз.
            Evgenii Legotckoi
            • 9. Januar 2018 07:40

            Я, конечно, могу ошибаться насчёт этого кода, и по идее выполнение должно завершаться сразу, когда вызван слот terminate, но в рамках многопоточных приложений может быть некоторая конкуренция между выполнениями кода и лучше перестраховаться.

              Ruslan Polupan
              • 9. Januar 2018 08:12

              Спасибо за советы. буду пробовать.

                Ruslan Polupan
                • 18. Januar 2018 06:51

                Все оказалось проще. Документацию то читал, но забыл нюансы.

                Не указывал в connect  тип подключения Qt::DirectConnection

                  Kommentare

                  Nur autorisierte Benutzer können Kommentare posten.
                  Bitte Anmelden oder Registrieren
                  Letzte Kommentare
                  A
                  ALO1ZE19. Oktober 2024 18:19
                  Fb3-Dateileser auf Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                  ИМ
                  Игорь Максимов5. Oktober 2024 17:51
                  Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                  d
                  dblas55. Juli 2024 21:02
                  QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                  k
                  kmssr9. Februar 2024 05:43
                  Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                  Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                  Jetzt im Forum diskutieren
                  J
                  JacobFib17. Oktober 2024 13:27
                  добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
                  JW
                  Jhon Wick2. Oktober 2024 01:52
                  Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
                  КГ
                  Кирилл Гусарев27. September 2024 19:09
                  Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
                  F
                  Fynjy22. Juli 2024 14:15
                  при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

                  Folgen Sie uns in sozialen Netzwerken