Ruslan Polupan
Ruslan PolupanJan. 9, 2018, 4:45 a.m.

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

потоки, 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;
}

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

ЗЫ: Программирование это для меня небольшое хобби. Так что буду рад любым комментариям.
We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

8
Evgenii Legotckoi
  • Jan. 9, 2018, 4:53 a.m.
  • (edited)

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

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

    Ruslan Polupan
    • Jan. 9, 2018, 6:31 a.m.

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

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

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

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


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

          Evgenii Legotckoi
          • Jan. 9, 2018, 7:38 a.m.

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

          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
            • Jan. 9, 2018, 7:40 a.m.

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

              Ruslan Polupan
              • Jan. 9, 2018, 8:12 a.m.

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

                Ruslan Polupan
                • Jan. 18, 2018, 6:51 a.m.

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

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

                  Comments

                  Only authorized users can post comments.
                  Please, Log in or Sign up
                  AD

                  C ++ - Test 004. Pointers, Arrays and Loops

                  • Result:50points,
                  • Rating points-4
                  m

                  C ++ - Test 004. Pointers, Arrays and Loops

                  • Result:80points,
                  • Rating points4
                  m

                  C ++ - Test 004. Pointers, Arrays and Loops

                  • Result:20points,
                  • Rating points-10
                  Last comments
                  Evgenii Legotckoi
                  Evgenii LegotckoiOct. 31, 2024, 2:37 p.m.
                  Django - Lesson 064. How to write a Python Markdown extension Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                  A
                  ALO1ZEOct. 19, 2024, 8:19 a.m.
                  Fb3 file reader on Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                  ИМ
                  Игорь МаксимовOct. 5, 2024, 7:51 a.m.
                  Django - Lesson 064. How to write a Python Markdown extension Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                  d
                  dblas5July 5, 2024, 11:02 a.m.
                  QML - Lesson 016. SQLite database and the working with it in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                  k
                  kmssrFeb. 8, 2024, 6:43 p.m.
                  Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                  Now discuss on the forum
                  m
                  moogoNov. 22, 2024, 7:17 a.m.
                  Mosquito Spray System Effective Mosquito Systems for Backyard | Eco-Friendly Misting Control Device & Repellent Spray - Moogo ; Upgrade your backyard with our mosquito-repellent device! Our misters conce…
                  Evgenii Legotckoi
                  Evgenii LegotckoiJune 24, 2024, 3:11 p.m.
                  добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                  t
                  tonypeachey1Nov. 15, 2024, 6:04 a.m.
                  google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                  NSProject
                  NSProjectJune 4, 2022, 3:49 a.m.
                  Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…

                  Follow us in social networks