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 р. 02: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 и т.д, а если память была освобождена, то куда это всё будет помещено? Никуда, пропадёт. Поэтому лучше держать базу данных за фассадом-обёрткой, чтобы всегда была выделена память под объект базы данных. В моём варианте с обёрткой, память выделенная под базу данных существует до конца работы программы.

        Коментарі

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

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

        • Результат:50бали,
        • Рейтинг балів-4
        m
        • molni99
        • 26 жовтня 2024 р. 01:37

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

        • Результат:80бали,
        • Рейтинг балів4
        m
        • molni99
        • 26 жовтня 2024 р. 01:29

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

        • Результат:20бали,
        • Рейтинг балів-10
        Останні коментарі
        ИМ
        Игорь Максимов22 листопада 2024 р. 11:51
        Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
        Evgenii Legotckoi
        Evgenii Legotckoi31 жовтня 2024 р. 14:37
        Django - Урок 064. Як написати розширення для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
        A
        ALO1ZE19 жовтня 2024 р. 08:19
        Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
        ИМ
        Игорь Максимов05 жовтня 2024 р. 07:51
        Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
        d
        dblas505 липня 2024 р. 11:02
        QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
        Тепер обговоріть на форумі
        Evgenii Legotckoi
        Evgenii Legotckoi24 червня 2024 р. 15:11
        добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
        t
        tonypeachey115 листопада 2024 р. 06:04
        google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
        NSProject
        NSProject04 червня 2022 р. 03:49
        Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
        9
        9Anonim25 жовтня 2024 р. 09:10
        Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

        Слідкуйте за нами в соціальних мережах