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
                  i
                  innorwallNov. 12, 2024, 6:12 a.m.
                  Django - Tutorial 055. How to write auto populate field functionality Freckles because of several brand names retin a, atralin buy generic priligy
                  i
                  innorwallNov. 12, 2024, 2:23 a.m.
                  QML - Tutorial 035. Using enumerations in QML without C ++ priligy cvs 24 Together with antibiotics such as amphotericin B 10, griseofulvin 11 and streptomycin 12, chloramphenicol 9 is in the World Health Organisation s List of Essential Medici…
                  i
                  innorwallNov. 11, 2024, 11:50 p.m.
                  Qt/C++ - Lesson 052. Customization Qt Audio player in the style of AIMP It decreases stress, supports hormone balance, and regulates and increases blood flow to the reproductive organs buy priligy online safe Promising data were reported in a PDX model re…
                  i
                  innorwallNov. 11, 2024, 10:19 p.m.
                  Heap sorting algorithm The role of raloxifene in preventing breast cancer priligy precio
                  i
                  innorwallNov. 11, 2024, 9:55 p.m.
                  PyQt5 - Lesson 006. Work with QTableWidget buy priligy 60 mg 53 have been reported by Javanovic Santa et al
                  Now discuss on the forum
                  i
                  innorwallNov. 12, 2024, 4:56 a.m.
                  добавить qlineseries в функции buy priligy senior brother Chu He, whom he had known for many years
                  i
                  innorwallNov. 11, 2024, 6:55 p.m.
                  Всё ещё разбираюсь с кешем. priligy walgreens levitra dulcolax carbs The third ring was found to be made up of ultra relativistic electrons, which are also present in both the outer and inner rings
                  9
                  9AnonimOct. 25, 2024, 4:10 p.m.
                  Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

                  Follow us in social networks