D
Dragon00119 апреля 2017 г. 17:29

Ошибка соединения

Здравствуйте. Делаю программу, в которой идет сначала авторизация в БД по логин/паролю, после чего, относительно того какая роль, открывает определенно окно, после открытия окна(для Администратора) производится запись в TableView. Если писать соединение сразу в main.cpp, то все прекрасно отображает,но у меня идет соединение через форму enterform. Если произвожу авторизацию через нее, то возникает такое окошко(Driver not loaded) и соответственно выборка не производится, но открывает нужное окно, где есть кнопка "Найти", после нажатия на которую, производится выборка. Я так понимаю, авторизация происходит как бы позже. Тогда возникает такой вопрос? Как это сделать правильно? Думал использовать QDialog,но как возвратит строку обратно в main.cpp? Или возможно ли не использовать QMainWindow(если допустим не нужны меню и прочее?) или он обязателен?

enterform::enterform(QWidget *parent) :
    QDialog(parent),
    ui(new Ui::enterform)
{
    ui->setupUi(this);
    //AdminForm = new AdministrationForm();
    QSettings settings("SOK", "Connection");
    ui->savecheckbox->setChecked(settings.value("saveSetting", false).toBool());
    if (ui->savecheckbox->isChecked())
    {
        QString user=settings.value("user").toString();
        QString pass=settings.value("pass").toString();
        ui->login->setText(user);
        ui->password->setText(pass);
    }
}

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

void enterform::on_EnterButton_clicked(){
    
    AdminForm = new AdministrationForm();
    QSettings settings("SOK", "Connection");
    QString host=settings.value("host").toString();
    int port=settings.value("port").toInt();
    QString base=settings.value("base").toString();
    QSqlDatabase db=QSqlDatabase::addDatabase("QPSQL");

    db.setHostName(host);
    db.setPort(port);
    db.setDatabaseName(base);
    db.setUserName(ui->login->text());
    db.setPassword(ui->password->text());


    if (db.open())
    {
        QSqlQuery query;
        if (ui->savecheckbox->isChecked()){

            settings.setValue("saveSetting", true);
            settings.setValue("user", ui->login->text());
            settings.setValue("pass",ui->password->text());

            query.exec("SELECT r.rolname FROM pg_catalog.pg_auth_members m, pg_catalog.pg_roles r WHERE m.roleid=r.oid and m.member =(SELECT oid FROM pg_catalog.pg_roles WHERE rolname = '"+ui->login->text()+"');");
            while (query.next())
            {
                QString rolname = query.value(0).toString();
                if (rolname == "Administation")
                   {
                    this->close();
                    AdminForm->showMaximized();

                   }
                else if(rolname == "director")
                    {
                    qDebug() << "test";
                    }
             }
        }
        else
        {
          settings.setValue("saveSetting", false);
          settings.setValue("user", ui->login->text());
          this->close();
        }
    }
    else
    {
        QMessageBox::warning(0, QObject::tr("Error"), db.lastError().databaseText());
    }


}

#include "administrationform.h"
#include "ui_administrationform.h"
#include <QTableView>
#include <QSqlError>
#include <QMessageBox>

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

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

void AdministrationForm::AllClientSelect()
{
    allClienttable = new QSqlQueryModel(this);
    allClienttable->setQuery("SELECT nomer_abonementa,familiya,imya,otchestvo,end_ref,mobile FROM klient");
      if (allClienttable->lastError().isValid())
      {
          QMessageBox::information(this, tr("TestErr"), allClienttable->lastError().databaseText());
      }
    allClienttable->setHeaderData(0, Qt::Horizontal, tr("Номер Абонемента"));
    allClienttable->setHeaderData(1, Qt::Horizontal, tr("Фамилия"));
    allClienttable->setHeaderData(2, Qt::Horizontal, tr("Имя"));
    allClienttable->setHeaderData(3, Qt::Horizontal, tr("Отчество"));
    allClienttable->setHeaderData(4, Qt::Horizontal, tr("Конец справки"));
    allClienttable->setHeaderData(5, Qt::Horizontal, tr("Телефон"));

    ui->allClient->setModel(allClienttable);
    ui->allClient->setSelectionBehavior(QAbstractItemView::SelectRows);
    ui->allClient->setColumnHidden(0, true);
    ui->allClient->setEditTriggers(QAbstractItemView::NoEditTriggers);
}

void AdministrationForm::on_pushButton_5_clicked()
{
        allClienttable = new QSqlQueryModel(this);
        allClienttable->setQuery("SELECT nomer_abonementa,familiya,imya,otchestvo,end_ref,mobile FROM klient");

        allClienttable->setHeaderData(0, Qt::Horizontal, tr("Номер Абонемента"));
        allClienttable->setHeaderData(1, Qt::Horizontal, tr("Фамилия"));
        allClienttable->setHeaderData(2, Qt::Horizontal, tr("Имя"));
        allClienttable->setHeaderData(3, Qt::Horizontal, tr("Отчество"));
        allClienttable->setHeaderData(4, Qt::Horizontal, tr("Конец справки"));
        allClienttable->setHeaderData(5, Qt::Horizontal, tr("Телефон"));
        ui->allClient->setModel(allClienttable);
        //Разрешаем выделение строк
        ui->allClient->setSelectionBehavior(QAbstractItemView::SelectRows);
        ui->allClient->resizeColumnsToContents();
        ui->allClient->setEditTriggers(QAbstractItemView::NoEditTriggers);

}

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

3
Evgenii Legotckoi
  • 20 апреля 2017 г. 2:37

У вас путаница с последовательностью действий. Нужно сделать авторизацию. И передать дальше управление в другое окно. Но при этом сам объект базы данных существует локально в методе on_EnterButton_clicked() . db сохраняется в стеке и не существует в другом окне, скорее всего соединение падает при окончании работы метода on_EnterButton_clicked() . Лучше делать обёртку, для работы с подключением к базе данных. Создать объект этой обертки к базе данных и через метод подключения передать по сигналам данные для авторизации. Также стоит выделить память через new для этого объекта обёртки. Класс обёртки у меня есть в следующей статье . Там есть метод connectToDataBase() , можете его переделать, чтобы он принимал в качестве аргументов пароль, логин и т.д. А возвращал например роль. Что касается передачи информации между окнами, то можете посмотреть вот этот пример . Там окна взаимодействуют через сигналы и слоты, как раз то, что нужно, чтобы вернуть данные из кона авторизации в главное окно.

Что касается использования QMainWindow, то можно использовать в качестве окна даже простой QWidget, но QDialog имеет уже встроенные стандартные кнопки.

Вообще, сам вот этот код должен быть в неком ядре программы, а не в конкретном окне.

QSqlDatabase db=QSqlDatabase::addDatabase("QPSQL");

db.setHostName(host);
db.setPort(port);
db.setDatabaseName(base);
db.setUserName(ui->login->text());
db.setPassword(ui->password->text());

db.open()

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

    D
    • 20 апреля 2017 г. 14:52

    Я нашел ошибку у себя, у меня получалось, что я создавал объект main окно до соединения с БД, из-за этого писало ошибку, переместил строку AdminForm = new AdministrationForm(); и все заработало, но перед этим сделал как вы просили, сделал отдельный класс для соединения с бд,в которую передавал значения и принимал. Я так понимаю верным должен быть 2 вариант, но почему? Разве объект БД не будет существовать до окончания программы?

      Evgenii Legotckoi
      • 20 апреля 2017 г. 23:45
      • Ответ был помечен как решение.

      Когда вы вызываете данный метод:

      QSqlDatabase db=QSqlDatabase::addDatabase("QPSQL");

      Вы сохраняете объект базы данных в стеке, то есть объект существует локально внутри метода on_EnterButton_clicked . А что происходит со всеми локальными переменными, когда метод заканчивает свою работу? Правильно, они уничтожаются. А вместе с этим уничтожением закрывается соединение с базой данных. Почему оно закрывается? Потому что локальный объект уничтожен и память освобождена. Хорошо, что ещё Segmentation Fault не словили.

      Конечно, в случае с QSqlDataBase всё обстоит хитрее, там несколько сложнее всё сделано внутри, поскольку сохраняются имена соединений, но, как я считаю, работать базаданных будет до тех пор, пока под неё выделена память. В конце концов QSqlDataBase хранит логин, пароль, порт, IP и т.д, а если память была освобождена, то куда это всё будет помещено? Никуда, пропадёт. Поэтому лучше держать базу данных за фассадом-обёрткой, чтобы всегда была выделена память под объект базы данных. В моём варианте с обёрткой, память выделенная под базу данных существует до конца работы программы.

        Комментарии

        Только авторизованные пользователи могут публиковать комментарии.
        Пожалуйста, авторизуйтесь или зарегистрируйтесь
        d
        • dsfs
        • 26 апреля 2024 г. 14:56

        C++ - Тест 004. Указатели, Массивы и Циклы

        • Результат:80баллов,
        • Очки рейтинга4
        d
        • dsfs
        • 26 апреля 2024 г. 14:45

        C++ - Тест 002. Константы

        • Результат:50баллов,
        • Очки рейтинга-4
        d
        • dsfs
        • 26 апреля 2024 г. 14:35

        C++ - Тест 001. Первая программа и типы данных

        • Результат:73баллов,
        • Очки рейтинга1
        Последние комментарии
        k
        kmssr9 февраля 2024 г. 5:43
        Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
        АК
        Анатолий Кононенко5 февраля 2024 г. 12:50
        Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
        EVA
        EVA25 декабря 2023 г. 21:30
        Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
        J
        JonnyJo25 декабря 2023 г. 19:38
        Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
        G
        Gvozdik19 декабря 2023 г. 8:01
        Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
        Сейчас обсуждают на форуме
        Evgenii Legotckoi
        Evgenii Legotckoi3 мая 2024 г. 0:07
        Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.
        IscanderChe
        IscanderChe30 апреля 2024 г. 14:22
        Во Flask рендер шаблона не передаётся в браузер Доброе утро! Имеется вот такой шаблон: <!doctype html><html> <head> <title>{{ title }}</title> <link rel="stylesheet" href="{{ url_…
        G
        Gar22 апреля 2024 г. 15:46
        Clipboard Как скопировать окно целиком в clipb?
        Павел Дорофеев
        Павел Дорофеев14 апреля 2024 г. 12:35
        QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
        f
        fastrex4 апреля 2024 г. 14:47
        Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

        Следите за нами в социальных сетях