Реклама
  • EVILEG
  • Ответ
  • 10 января 2018 г. 7:30

Запрос к базе в отдельном потоке.

Я бы тогда передал вектор некоторой информации, классов, которые будут отвечать за строки в таблице.


class TableInfo
{
public:
    /* Секция геттеров и сеттеров */
  
private:
    int m_id;
    int m_info_1;
    int m_info_2;
    QString m_description;  
}
В самом потоке информацию можно получить через QSqlQuery. Потом пройтись в цикле и подготовить вектор этих элементов
QVector<TableInfo> infoVector;
Сам вектор можно передавать через сигнал слотовое соединение, как сделано вот в этой статье про moveToThread . Думаю, что Вы её уже видели.

Единственное, только нужно будет обезопаситься мьютексами во время передачи информации между потоками.
  • Ruslan
  • Вопрос
  • 9 января 2018 г. 13:34

Запрос к базе в отдельном потоке.

поток, запрос, модель

В связи с тем что запросы к базе выполняется достаточно долго (удаленные базы на объектах с низкой скоростью интернет соединения) запросы хочется выполнять в отдельных потоках.

Вопрос в следующем: Возможно ли получать из потока модель(QSqlQueryModel, QAbstractItemModel или какую либо другую).
Если нет, то каким образом лучше всего передавать результат работы запроса в основной поток для дальней шей обработки.
Данные нужны только для чтения.
  • EVILEG
  • Ответ
  • 9 января 2018 г. 12: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() всегда вызывался один раз.
  • Ruslan
  • Вопрос
  • 9 января 2018 г. 9: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;
}

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

ЗЫ: Программирование это для меня небольшое хобби. Так что буду рад любым комментариям.

Десятичный разделитель

Здравствуйте! Подскажите, как правильно осуществлять ввод/вывод в интерфейсе вещественных чисел с разделителем?

Ситуация такова. Есть sqlite база и таблицы с полями типа REAL. В них числа записаны с запятой (символы разделителя). В интерфейс вывод согласно вашим урокам через mapper, но в роли разделителя получается точка.
При записи данных из TextField в эти поля использую:
quer.bindValue(":normed",       data2[5].toString()=="" ? QVariant(QVariant::String):data2[5].toReal());
Так вот, если при вводе в качестве разделителя использую запятую, то в базу пишется 0. Если ввод через точку, то в базу записывается через запятую. Из документации известно, что 0 возвращается при неудачном преобразовании строки в число. Установки QLocale после вызова QApplication app(argc, argv) результата не дают, на многих форумах пишут что проще менять точку на запятую типа:
str.replace(".", ",");
В общем очень запутался и хотелось бы узнать как правильно справляться с этой проблемой.
  • EVILEG
  • Ответ
  • 26 декабря 2017 г. 20:45

Как вытащить данные из трех таблиц MySQL

При создании пользователя создавайте запись с его балансом по принципу OneToOne связи.

Это будет правильным подходом, поскольку одна запись баланса будет привязана к одному пользователю в любом случае. Входящие транзакции в кассу должны изменять баланс. Также можно автоматически изменять баланс и при добавлении работ.
Иначе запросы к базе будут настолько страшными и огромными, что база с трудом будет справляться с простейшими операциями.
Да и нужно быть Гуру SQL. Я к сожалению затрудняюсь написать обработку ситуации при отсутствии каких-либо записей в базе данных.
  • EVILEG
  • Ответ
  • 26 декабря 2017 г. 13:13

Как вытащить данные из трех таблиц MySQL

Чтобы вывести в таблицу такие данные, вам нужно будет использовать QSqlQueryModel .

А запрос должен выглядеть следующим образом:
SELECT id_user, SUM(summ_work)
FROM table_works
GROUP BY id_user
Следующий запрос должен выбрать колонки id_user и summ_work из таблицы выполненных работ и при этом суммировать все значения долга работ по пользователям, как раз с учётом группировки по id_user.

Единственное, здесь будет подставлен id пользователя, а не его имя. Для этого потребуется писать более сложный join запрос. Но реализуйте для начала этот вариант.
  • pro100belik
  • Вопрос
  • 26 декабря 2017 г. 12:52

Как вытащить данные из трех таблиц MySQL

Есть три таблицы

1. Клиенты
2. Выполненные работы
3. Касса

Структура первой таблицы id_user, name_user
Структура второй таблицы id_work, id_user, name_work, summ_work
Структура третей таблицы id_cash, id_user, summ_cash

Как правильно рассчитать задолженность клиентов, и вывести их список? Подскажите пожалуйста.


Изменение данных в таблицах базы

Извиняюсь за заданный вопрос. Сам разобрался. Создал следующие методы:

    bool editTableBaza(const QVariantList &data);
    bool editTableBaza(const QString &Baza_id, const QString &Bazakks);
Их конструкторы:
bool DataBase::editTableBaza(const QVariantList &data)
{    
    QSqlQuery querys;    
    querys.prepare("update Baza set KKS = :Bazakks where Baza.id = :Baza_id");
    querys.bindValue(":Baza_id", data[0].toString());
    querys.bindValue(":Bazakks", data[1].toString());
    qDebug()<<data[0].toString();
    if(!querys.exec()){
        qDebug() << "error update Baza ";
        qDebug() << querys.lastError().text();
        return false;
    } else {
        return true;
    }
    return false;
}

bool DataBase::editTableBaza(const QString &Baza_id, const QString &Bazakks)
{
    QVariantList data;
    data.append(Baza_id);
    data.append(Bazakks);
    if(editTableBaza(data))
        return true;
    else
        return false;
}
а в QML вызываем:
database.editTableBaza(stackView.baza_id, tf_kks.text) //stackView.baza_id содержит id изменяемой записи
В общем все по аналогии с созданием новой записи из урока.

Изменение данных в таблицах базы

Добрый день! Сейчас работаю с sqlite базой данных на основе ваших уроков. Выборка данных и их представление, удаление и создание новых работают без проблем и все красиво. Но как быть с редактированием данных таблиц базы? В ваших уроках этот момент почему-то опущен. Подскажите, в какую сторону копать?

P.S. Таблицы (самодельные ListView) заполняю через модели и роли, странички с данными объектов посредством вашего Data Mapper через mapper.addMapping() (почему-то мне кажется это очень удобным).
Реклама
  • falcon
  • 16 января 2018 г. 17:25

Qt - Тест 001. Сигналы и слоты

  • Результат 100 баллов
  • Очки рейтинга 10
  • falcon
  • 16 января 2018 г. 17:22

Qt - Тест 001. Сигналы и слоты

  • Результат 68 баллов
  • Очки рейтинга -1
  • falcon
  • 16 января 2018 г. 17:18

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

  • Результат 73 баллов
  • Очки рейтинга 1
Последние комментарии

QML - Урок 021. Переключение между окнами в QML

Спасибо всем. Все получилось. Прикручиваю логику.

  • BlinCT
  • 14 января 2018 г. 19:28

Разработка на Qt под iOS

Вот честно, на сколько же муторно под огрызок что то делать. Куча проблем) А вод линь или под Андроид все просто и тривиально))

  • folax
  • 12 января 2018 г. 9:16

QML - Урок 021. Переключение между окнами в QML

Ничего сложного, делаете по тех заданию 3 файла qml, называете их как указанно в тех задании, потом из первого окна через Loader их переключаете, в окне 2 и 3 делаете сигналы которые при закры...

QML - Урок 021. Переключение между окнами в QML

Все верно, я и не говорил что этот кусок кода лично мое произведение. Это тоже верно: Это задание для прохождения на собеседование в одну из крупных украинских IT компаний. Логику ...

  • folax
  • 12 января 2018 г. 8:13

QML - Урок 021. Переключение между окнами в QML

int main(int argc, char *argv[]){ QApplication app(argc, argv); Logic logic; QQmlApplicationEngine engine; engine.rootContext()->setContextProperty("logic", &logic)...

Сейчас обсуждают на форуме

Как проверить доступность сервера

Точно!!! Я сейчас так пробую QNetworkReply *replay_news = networkManager_news->get(QNetworkRequest(QUrl(url_news)));connect(networkManager_news, &QNetworkAccessManager::...

ChartView. Отображение метки данных точки серии при наведении курсора

Спасибо большущее за советы! Все получилось через ScatterSeries. Методы remove() как-то сходу не дались, удаляет в первый раз, а потом программа падает... Не стал тратить время и воспользовалс...

QGraphicsScene

спасибо, за подробное объяснение строчки, а с зумом я разобрался, все работает

  • EVILEG
  • 15 января 2018 г. 17:21

Qt webgl

Насчёт проверки подключения клиента я не в курсе. Что касается экземпляров приложения, то из того, что я читал получается, что нет необходимости в нескольких экземплярах для нескольких кл...

  • EVILEG
  • 15 января 2018 г. 11:39

Проблема добавления #DEFINE при сборке CMak'ом

А Вы не пробовали сделать предкомпилированные библиотеки boost под свою систему, а потом уже подключать собранные библиотеки Boost`а? Просто один только boost может собираться на пару гиг...