Ruslan Polupan
Ruslan PolupanСәуір 17, 2019, 3:58 Т.Ж.

Потоки. Параллельные действия.

QTableWidget, QThread, threads

Доброго времени суток!

Нужен совет по много поточности.
Есть вот такая вот табличка.

Столбец статуc хотелось бы получать в потоке для всех записей (АЗС).
Для одной записи я реализовал получение статуса сервера базы данных.

Вопрос в том как делать это для всех записей, количество записей до 200 чтобы для каждой записи это делалось в потоке.


Реализация получения статуа в потоке

CheckAzsStatus *checkStatus = new CheckAzsStatus(modelConnections->data(modelConnections->index(0,0,QModelIndex())).toInt(),
                                                     modelConnections->data(modelConnections->index(0,1,QModelIndex())).toString());
    QThread *thread = new QThread(this);

    checkStatus->moveToThread(thread);


    connect(thread,&QThread::started, this, &ShowPage::slotStartExecute,Qt::DirectConnection);
    connect(thread,&QThread::started, checkStatus, &CheckAzsStatus::slotCheckAzsStatus,Qt::DirectConnection);


    connect(checkStatus,&CheckAzsStatus::signalSendResult,this,&ShowPage::slotGetAzsStatus);

    connect(checkStatus,&CheckAzsStatus::finished,this,&ShowPage::slotStopExecute);
    connect(checkStatus,&CheckAzsStatus::finished,thread,&QThread::quit);
    connect(checkStatus,&CheckAzsStatus::finished,checkStatus,&CheckAzsStatus::deleteLater);
    connect(thread,&QThread::finished, thread, &QThread::deleteLater);

    thread->start();


}


void ShowPage::slotStartExecute()
{
    qInfo(logInfo()) << Q_FUNC_INFO << QDateTime::currentDateTime().toString("dd-MM-yyyy hh:mm:ss.zzz") <<  "Thread started";
}

void ShowPage::slotStopExecute()
{
    qInfo(logInfo()) << Q_FUNC_INFO << QDateTime::currentDateTime().toString("dd-MM-yyyy hh:mm:ss.zzz") <<  "Thread stoped";
}

void ShowPage::slotGetAzsStatus(bool res)
{
    if(res){
        ui->tableWidget->item(0,2)->setText("ON LINE");
    } else {
        ui->tableWidget->item(0,2)->setText("OFF LINE");
    }

}

Реализация проверки
checkazsstatus.h

#ifndef CHECKAZSSTATUS_H
#define CHECKAZSSTATUS_H

#include <QObject>

class CheckAzsStatus : public QObject
{
    Q_OBJECT
public:
    explicit CheckAzsStatus(int term, QString ip, QObject *parent = nullptr);

signals:
    void signalSendResult(bool status);
    void finished(int term);

public slots:
    void slotCheckAzsStatus();
private:
    QString m_serverName;
    int m_terminalID;
};

#endif // CHECKAZSSTATUS_H

checkazsstatus.cpp

#include "checkazsstatus.h"
#include "LoggingCategories/loggingcategories.h"
#include <QTcpSocket>

CheckAzsStatus::CheckAzsStatus(int term, QString ip, QObject *parent) : QObject(parent)
{
    m_serverName = ip;
    m_terminalID = term;
    qInfo(logInfo()) << "IP" << m_serverName << "Terminal" << m_terminalID;
}

void CheckAzsStatus::slotCheckAzsStatus()
{

    bool status;
    QTcpSocket *tcpSocket = new QTcpSocket();
    tcpSocket->connectToHost(m_serverName, 3050);
    if(tcpSocket->waitForConnected(30000)){
        status = true;
    } else {
        status = false;
    }
    emit signalSendResult(status);
    emit finished(m_terminalID);
}

Буду благодарен за любою помошь.

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

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

25
Алексей Внуков
  • Сәуір 17, 2019, 4:03 Т.Ж.

как то немного сомбурно выглядит описание. вы хотите получать статус 200 обьектов в отдельном потоке или для каждого свой поток? я так понял описание обьектов находится в бд?

    Ruslan Polupan
    • Сәуір 17, 2019, 4:15 Т.Ж.

    Простите за сумбурность. :-)
    Да для каждого объекта в своем потоке, поскольку связь с объектами оставляет желать лучшего.
    При наличии связи с обектом на базе АЗС выполняется SQL запрос на получении информации ( в данном случае наименование видов топлива на АЗС)
    Описание объектов в объекте QSqlTableModel, получаемой из базы данных.

      Алексей Внуков
      • Сәуір 17, 2019, 4:36 Т.Ж.

      как вариант (работать буде при неограниченном колличестве обьектов). добавьте в класс проверки статуса, полную реализацию проверки с сигналами и тд, только в конектах можно завязать не на слот класа а на новый сигнал, после в основном классе, можно сделать запрос в БД на получение данных о ваших обьектах (ID, IP и тд), и при разборе этих данных создавайте обьект проверки и сделайте перепривязку сигналов, при создании можете его пихнуть в вектор например, и перебором вектора пихать каждый обьект в отдельный поток для проверки(например QtConcurrent), а полученый результат через сигналы парсить

        nayk1982
        • Сәуір 17, 2019, 4:39 Т.Ж.

        Придется в каждом потоке создавать свое соединение с БД: документация Qt

          Алексей Внуков
          • Сәуір 17, 2019, 4:50 Т.Ж.
          • (өңделген)

          а смысл? соеденение с БД можно сделать глобальным и в потоке при запросе просто делать указание QSqlQuery query(str,db) документация

            Ruslan Polupan
            • Сәуір 17, 2019, 5:01 Т.Ж.

            Алексей, данные об объектах я получаю.

                modelConnections = new QSqlQueryModel();
            //    QSqlDatabase db = QSqlDatabase::database();
                QString inStr = m_listTerminals.join(",");
            
            
                QString strSQL = QString("SELECT c.TERMINAL_ID, c.SERVER_NAME, c.DB_NAME, c.CON_PASSWORD from CONNECTIONS c "
                                         "WHERE c.TERMINAL_ID IN (%1) and c.CONNECT_ID = 2 "
                                         "ORDER BY c.TERMINAL_ID")
                        .arg(inStr);
                qInfo(logInfo()) << "SQL" << strSQL;
                modelConnections->setQuery(strSQL);
            
                qInfo(logInfo()) << "Model Row Count" << modelConnections->rowCount();
            
                ui->tableWidget->clearContents();
                ui->tableWidget->clear();
            
                ui->tableWidget->setColumnCount(COLUMN_COUNT);
                ui->tableWidget->setHorizontalHeaderLabels(TABLE_COLUMN_LABELS);
                ui->tableWidget->horizontalHeader()->setStretchLastSection(true);
                ui->tableWidget->verticalHeader()->hide();
            
                for(int i = 0; i<modelConnections->rowCount();++i){
                    ui->tableWidget->insertRow(i);
                    ui->tableWidget->setItem(i,0,new QTableWidgetItem(modelConnections->data(modelConnections->index(i,0,QModelIndex())).toString()));
                    ui->tableWidget->setItem(i,1,new QTableWidgetItem(modelConnections->data(modelConnections->index(i,1,QModelIndex())).toString()));
                    ui->tableWidget->setItem(i,2, new QTableWidgetItem("Проверка"));
                }
            
                ui->tableWidget->resizeColumnsToContents();
            

            Вопрос как раз в реализации. Перечитав кучу доков и форумов я только окончательно запутался. А хотелось бы просто посмотреть кусок рабочего кода. :-)

              nayk1982
              • Сәуір 17, 2019, 5:02 Т.Ж.

              Это если в одном потоке. В параллельных потоках надо создавать соединение, причем с разными именами, пример:

              QSqlDatabase::addDatabase("QMYSQL", "db_thread1");
              
                Ruslan Polupan
                • Сәуір 17, 2019, 5:02 Т.Ж.

                Это я в курсе, мне в потоке не надо пока подключатся к БД.

                  Ruslan Polupan
                  • Сәуір 17, 2019, 5:04 Т.Ж.

                  Так не даст подключится. Для потока создается отдельный пулл подключений к БД. я с этим уже сталкивался.

                    Алексей Внуков
                    • Сәуір 17, 2019, 5:11 Т.Ж.
                    • (өңделген)

                    проверено на практике - делаем глобальный

                    global.h

                    extern QSqlDatabase db;
                    

                    global.cpp

                    QSqlDatabase db;
                    

                    в основном классе

                     db=QSqlDatabase::addDatabase("QDB");
                     db.setDatabaseName(db_name);
                     db.setUserName(db_login);
                     db.setHostName(db_host);
                     db.setPassword(db_password)
                    

                    и потом в любом месте в любом потоке через

                    QSqlQuery query(str,db);
                    

                    имеем доступ к единственному экземпляру БД, и больше ничего создавать не нужно

                      Алексей Внуков
                      • Сәуір 17, 2019, 5:21 Т.Ж.
                      • (өңделген)

                      примерно так (кусок реализации работы с системами Satel)
                      soket.h

                      #ifndef SOCKET_H
                      #define SOCKET_H
                      
                      #include <QTcpSocket>
                      #include <QByteArray>
                      #include <QInputDialog>
                      
                      class socket : public QObject
                      {
                          Q_OBJECT
                      public:
                          socket(const QString name, const QString host, const int port, QWidget *parent = nullptr);
                          QTcpSocket *psocket;
                          QString name;//??
                          QString host;
                          int port;
                      
                          ~socket();
                      
                          void connect_to_tcp  ();
                      signals:
                          //void printmsg(QString);
                          void conect(socket* soc);
                          void slotReadyRead (socket* soc, QTcpSocket* tsoc);
                          void slotError (socket* soc, QTcpSocket* tsoc, QAbstractSocket::SocketError);
                          //void slotConnected   (socket* soc);
                      };
                      
                      

                      soket.cpp

                      #include "socket.h"
                      
                      socket::socket(const QString name, const QString host, const int port, QWidget *parent)
                      {
                          psocket=new QTcpSocket(this);
                          this->host=host;
                          this->port=port;
                          this->name=name;
                          connect(psocket, &QTcpSocket::connected, [this]{emit conect(this);});
                          connect(psocket, &QTcpSocket::readyRead, [this]{emit slotReadyRead(this,psocket);});
                          connect(psocket, QOverload<QAbstractSocket::SocketError>::of(&QAbstractSocket::error),
                                  [this](QAbstractSocket::SocketError socketError){emit slotError(this, psocket, socketError);});// SLOT(slotError(QAbstractSocket::SocketError)));
                      }
                      
                      socket::~socket()
                      {
                      
                      }
                      
                      void socket::connect_to_tcp()
                      {
                          psocket->connectToHost(host,port);
                      }
                      

                      в основном рабочем классе делал так

                       QSqlQuery query_m("SELECT name, ip_address, port FROM mod_satel");;
                              while (query_m.next()) {
                                  QString name=query_m.value(0).toString();
                                  QString host=query_m.value(1).toString();
                                  int i = query_m.value(2).toInt();
                                  //qDebug() << "Adding new soket";
                                  socket* p_socket=new socket(name,host,i);
                                  connect(p_socket,&socket::conect, this,&satel::slotConnected_satel);
                                  connect(p_socket,&socket::slotReadyRead,this,&satel::slotReadyRead_satel);
                                  connect(p_socket,&socket::slotError,this,&satel::slotError_satel);
                                  m_socket.append(p_socket);
                              }
                      

                      идею взял вот с этой книжки "Lazar G., Penea R. - Mastering Qt 5 - 2016"

                        Ruslan Polupan
                        • Сәуір 17, 2019, 5:25 Т.Ж.

                        Спасибо, но меня больше интересует реализация потоков.

                          Алексей Внуков
                          • Сәуір 17, 2019, 5:38 Т.Ж.
                          • (өңделген)

                          а что мешает потом сделать так? (при определенных условиях, отправляется в отдельный поток на обработку, количество совпадений по условию не предсказуемо)

                           if(vector.contains(N))
                          {
                              QtConcurrent::run(this, &satel::arm_disarm, buffer);
                          }
                          
                            Алексей Внуков
                            • Сәуір 17, 2019, 5:49 Т.Ж.
                            • (өңделген)

                            ну или попробывать такой вариант(ваш обьект в другом потоке останется жить пока не удалите)

                             QSqlQuery query_m("SELECT name, ip_address, port FROM mod_satel");;
                                    while (query_m.next()) {
                                        QString name=query_m.value(0).toString();
                                        QString host=query_m.value(1).toString();
                                        int i = query_m.value(2).toInt();
                                        //qDebug() << "Adding new soket";
                                        socket* p_socket=new socket(name,host,i);
                                        connect(p_socket,&socket::conect, this,&satel::slotConnected_satel);
                                        connect(p_socket,&socket::slotReadyRead,this,&satel::slotReadyRead_satel);
                                        connect(p_socket,&socket::slotError,this,&satel::slotError_satel);
                                       QThread *thread=new QThread(this);
                                       p_socket->moveToThread(thread);
                                       thread->start();
                                    }
                            
                              R
                              • Сәуір 17, 2019, 5:05 Т.Қ.
                              • (өңделген)

                              Добрий вечір, не рекомендував би вам використоувати для цих цілей, багато потоків, один потік це приблизно 67мб (принаймні було в моєму випадку) і того вам на 200 потоків потрібно не мало памяті,
                              моя реалізація стосується роботи з QWebSocket короткий код нижче, все перевірено і працює, + не потрібно робити багато конекшинів з кожного потоку до бази, і не так важко реалізувати комунікацію

                              QWebSocket *pSocket = m_pWebSocketServer->nextPendingConnection();
                              
                                      QString id_socket = QString("%1%2").arg(QDateTime::currentDateTime().toString("yyyyMMddhhmmsszzz")).arg("blablabla");
                              
                                      pSocket->setObjectName(id_socket);
                                      UserConnect* userThread = new UserConnect(pSocket);
                                      connect(pSocket, &QWebSocket::disconnected, this, &SSLWebSocketServer::socketDisconnected);
                                      connect(pSocket, &QWebSocket::disconnected, userThread, &UserConnect::deleteLater);
                              
                              // цей параметр з іншого класу, але дляприкладу привів
                                  QMap<QString, QWebSocket *> clients;
                              
                                  lients.insert(id_socket, pSocket);
                              
                              
                              
                              далі можна робити комунікації по ObjectName
                              
                              по типу
                              
                                     QWebSocket *pSocket = clients.value(id_socket);
                              
                                      if(pSocket)
                                      {
                                           pSocket->sendTextMessage(message);
                                      }
                              
                                u
                                • Сәуір 17, 2019, 10:20 Т.Қ.

                                добрый день, мне кажется при таком количестве одновременных запросов лучше работать через брокер

                                  Алексей Внуков
                                  • Сәуір 18, 2019, 2:21 Т.Ж.

                                  я не знаю как у вас получилось 1поток=67мб (может у вас был не поток а процесс?). у меня есть проект, в нем постоянно активно 28 потоков, с таймерами и бесконечными циклами, и они регулярно стучат в БД,так вот там у меня с запущенным интерфейсом (QML) занято 105м,, без интерфейса все 42мб

                                    Ruslan Polupan
                                    • Сәуір 18, 2019, 2:29 Т.Ж.
                                    • (өңделген)

                                    Большое спасибо за наводящие ответы. :-)
                                    QTcpSocket мне нужен всего лишь для проверки доступна ли удаленная база данных.
                                    Если доступна, то тогда у же к ней будем обращаться с запросами.

                                      Алексей Внуков
                                      • Сәуір 18, 2019, 2:33 Т.Ж.

                                      отпишитесь потом по результатам, что получилось сколько занимает, и какая нагрузка в общем

                                        Ruslan Polupan
                                        • Сәуір 18, 2019, 2:47 Т.Ж.

                                        Следующий этап это выполнение на каждой активной базе выполнять в потоке запрос

                                        select t.TANK_ID, f.NAME from tanks t
                                        left join FUELS f on f.FUEL_ID = t.FUEL_ID
                                        where t.ISACTIVE = 'T'
                                        

                                        Вот там и будем мерять.
                                        Потому что статистика выволнения не очень впечатляет :-)

                                        Executing statement...
                                        Statement executed (elapsed time: 0.000s).
                                        360 fetches, 19 marks, 7 reads, 16 writes.
                                        0 inserts, 0 updates, 0 deletes, 33 index, 14 seq.
                                        Delta memory: 200 bytes.
                                        Total execution time: 5.026s
                                        Script execution finished.

                                        Кстати вопрос на эту тему. QSqlQuery позволяет какимно образом получать такую информаци?

                                          Ruslan Polupan
                                          • Сәуір 18, 2019, 9:17 Т.Ж.

                                          Реаизовал с использованием QtConcurent, дабы получить сигнал что все потоки по опросу порта завершились.

                                              checkOnline = new QFutureWatcher<bool>(this);
                                              connect(checkOnline,&QFutureWatcher<QTcpSocket>::started, this, &ShowPage::slotStartExecute);
                                              connect(checkOnline,&QFutureWatcher<QTcpSocket>::resultReadyAt,this,&ShowPage::slotStopExecute);
                                              connect(checkOnline,&QFutureWatcher<QTcpSocket>::finished,this,&ShowPage::slotFinished);
                                          
                                              std::function<bool(QString)> getOnline = [] (const QString ipAZS){
                                                  QTcpSocket tcpSocket;
                                                  tcpSocket.connectToHost(ipAZS,3050);
                                                  return tcpSocket.waitForConnected(30000);
                                              };
                                          
                                              checkOnline->setFuture(QtConcurrent::mapped(m_listIP,getOnline));
                                          }
                                          
                                          
                                          void ShowPage::slotStartExecute()
                                          {
                                             qInfo(logInfo()) << Q_FUNC_INFO << QDateTime::currentDateTime().toString("dd-MM-yyyy hh:mm:ss.zzz") <<  "Thread started";
                                          }
                                          
                                          void ShowPage::slotStopExecute(int term)
                                          {
                                              if(checkOnline->resultAt(term)) {
                                                  ui->tableWidget->item(term, 2)->setText("ON LINE");
                                                  ui->tableWidget->item(term,2)->setTextColor("Green");
                                                  ui->tableWidget->item(term, 2)->setIcon(QIcon(":/Icons/connect.png"));
                                              } else {
                                                  ui->tableWidget->item(term, 2)->setText("OFF LINE");
                                                  ui->tableWidget->item(term, 2)->setTextColor("Red");
                                                  ui->tableWidget->item(term, 2)->setIcon(QIcon(":/Icons/disconnect.png"));
                                              }
                                          }
                                          
                                          void ShowPage::slotGetAzsStatus(bool res)
                                          {
                                              isOnline = res;
                                          }
                                          
                                          void ShowPage::slotFinished()
                                          {
                                              qInfo(logInfo()) << "QtConcurent finished";
                                          }
                                          
                                            Ruslan Polupan
                                            • Сәуір 18, 2019, 9:47 Т.Ж.

                                            отпишитесь потом по результатам, что получилось сколько занимает, и какая нагрузка в общем

                                            А как мерять :-)

                                              Алексей Внуков
                                              • Сәуір 19, 2019, 3:49 Т.Ж.

                                              просто скажите что по загрузке процессора и оперативы, когда происходит опрос

                                                Ruslan Polupan
                                                • Сәуір 19, 2019, 5:43 Т.Ж.

                                                Можно просмотреть результаты тут
                                                https://drive.google.com/file/d/1tFmrljPrxjqkb6OZSvCmF5uGtuHvlKPq/view?usp=sharing

                                                  R
                                                  • Сәуір 19, 2019, 5:55 Т.Ж.

                                                  мені важко це зараз навіть перевірити, тому що знайшов коміт, це ще було в 2016 році, і цей код не буде працювати коректно зараз, єдине скажу що це були QThread

                                                    Пікірлер

                                                    Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                                                    Кіріңіз немесе Тіркеліңіз
                                                    AD

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

                                                    • Нәтиже:50ұпай,
                                                    • Бағалау ұпайлары-4
                                                    m
                                                    • molni99
                                                    • Қаз. 26, 2024, 1:37 Т.Ж.

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

                                                    • Нәтиже:80ұпай,
                                                    • Бағалау ұпайлары4
                                                    m
                                                    • molni99
                                                    • Қаз. 26, 2024, 1:29 Т.Ж.

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

                                                    • Нәтиже:20ұпай,
                                                    • Бағалау ұпайлары-10
                                                    Соңғы пікірлер
                                                    ИМ
                                                    Игорь МаксимовҚар. 22, 2024, 11:51 Т.Ж.
                                                    Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                                                    Evgenii Legotckoi
                                                    Evgenii LegotckoiҚаз. 31, 2024, 2:37 Т.Қ.
                                                    Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                                                    A
                                                    ALO1ZEҚаз. 19, 2024, 8:19 Т.Ж.
                                                    Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                                    ИМ
                                                    Игорь МаксимовҚаз. 5, 2024, 7:51 Т.Ж.
                                                    Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                                    d
                                                    dblas5Шілде 5, 2024, 11:02 Т.Ж.
                                                    QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                                    Енді форумда талқылаңыз
                                                    m
                                                    moogoҚар. 22, 2024, 7:17 Т.Ж.
                                                    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 LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
                                                    добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                                                    t
                                                    tonypeachey1Қар. 15, 2024, 6:04 Т.Ж.
                                                    google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                                                    NSProject
                                                    NSProjectМаусым 4, 2022, 3:49 Т.Ж.
                                                    Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…

                                                    Бізді әлеуметтік желілерде бақылаңыз