М
Миша16. Dezember 2017 12:11

Пауза в многопоточности

Здравствуйте, можно ли поставить процесс на паузу при выполнении функций в многопоточности QtConcurrent::map?

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.

Magst du es? In sozialen Netzwerken teilen!

18
Evgenii Legotckoi
  • 16. Dezember 2017 12:23
  • (bearbeitet)

QFuture, который возвращается QtConcurrent::map имеет методы pause() и resume() и теоретически должен поддерживать этот функционал.


Но для QtConcurrent::run, например, в документации сказано, что пауза и возобновление для QFuture поддерживаться не будут. Однако для QtConcurrent::map такого не сказано.

Поэтому просто проверьте, будут ли работать те методы. Если будут работать, то всё ОК, если не будут, значит это также как и для QtConcurrent::run не поддерживается.
    М
    • 17. Dezember 2017 02:30

    Во  время работы поверх окна приложения активно окно QProgressDialog. И у него есть только одна кнопка. Код выглядит следующим образом. Можно ли добавить еще две кнопки или диалог баттон бокс в окно QProgressDialog? Или можно как то иначе произвести сигнал?

    QProgressDialog dialog;
    
           // Создать объект QFutureWatcher и подключить сигналы и слоты к диалогу.
           // Мониторинг прогресса изменений для future
           QFutureWatcher<void> futureWatcher;
           connect(&futureWatcher, &QFutureWatcher<void>::finished, &dialog, &QProgressDialog::reset);
           connect(&dialog, &QProgressDialog::canceled, &futureWatcher, &QFutureWatcher<void>::cancel);
           connect(&dialog, &QProgressDialog::canceled,[this](){StopStrategy=true;});
           connect(&futureWatcher, &QFutureWatcher<void>::progressRangeChanged, &dialog, &QProgressDialog::setRange);
           connect(&futureWatcher, &QFutureWatcher<void>::progressValueChanged, &dialog, &QProgressDialog::setValue);
    
    
    
           //QFuture<void> Perebor1 = QtConcurrent::map(Perebor,[this](const double& d){ StrategyCod2(d);});  //просчет стратегии в нескольхих потоках
           FileForRecord.open(QIODevice::WriteOnly);  //открывает файл на запись
    
           QLibrary MyLib("C:\\Qt\\project\\build-StrategyCod-Desktop_Qt_5_9_2_MinGW_32bit-Debug\\debug\\StrategyCod");  //путь к библиотеке
           MyLib.unload();//отключение библиотеки
           MyLib.load();//чтение
           typedef void (*MyPrototype)(double,MainWindow *);
           MyPrototype StrategyCod = (MyPrototype) MyLib.resolve("StrategyCod");
           futureWatcher.setFuture(QtConcurrent::map(Perebor,[=](const double& d){ StrategyCod(d,this);}));
           // Показать диалог и запустить программную петлю диалога
           dialog.exec();
    
           futureWatcher.waitForFinished();
           MyLib.unload();//отключение библиотеки

     

      Evgenii Legotckoi
      • 17. Dezember 2017 04:55

      Если честно, здесь уже проще самому написать нужный класс.

      Дело в том, что QProgressDialog не имеет QDialogButtonBox , как впрочем и QDialog . Да, в QProgressDialog есть кнопка отмена , но только она там и есть. А QDialog вообще не имеет кнопок и QDialogButtonBox`а .
      Но QDialog используется как база для создания своих диалоговых окон.
      Так что на мой взгляд, проще создать свой прогресс диалог и реализовать самостоятельно весь необходимый функционал. В конце концов есть и отдельный виджет прогресс бара, который можно использовать в собственном диалоге.
      Ещё можно, конечно, наследоваться от QProgressDialog`a, но скорее всего это будет несколько более усложнённым решением в плане вёрстки.
        М
        • 18. Dezember 2017 08:03

        У меня есть QProgressBar "ui->ProgressBar2".

        Можно ли ему аналогично передавать значение объема?
        Могу написать так
        connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);});
        Но как задать значение прогресса?
        Если пишу так, то сразу 10 ошибок
        connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});

          Evgenii Legotckoi
          • 18. Dezember 2017 08:11

          Киньте текст ошибок. Хотя у меня подозрение, что вы просто забыли/удалили #include <QDebug>

            М
            • 18. Dezember 2017 08:13
            C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
            connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});
            ^
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:185: ошибка: use of deleted function 'QFutureWatcherBase::QFutureWatcherBase(const QFutureWatcherBase&)'
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:454: ошибка: 'QObject::QObject(const QObject&)' is private
            Q_DISABLE_COPY(QObject)
            ^
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:55: ошибка: within this context
            class Q_CORE_EXPORT QFutureWatcherBase : public QObject
            ^
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:55: ошибка: use of deleted function 'QObject::QObject(const QObject&)'
            C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>::<lambda>(MainWindow::on_ButtonRunMulti_clicked()::<lambda()>&&)'
            connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});
            ^
            C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:307: ошибка: use of deleted function 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>::<lambda>(const MainWindow::on_ButtonRunMulti_clicked()::<lambda()>&)'
            return connect(sender, signal, sender, slot, Qt::DirectConnection);
            ^
            C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
            C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:313: ошибка: 'static typename std::enable_if<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, const QObject*, Func2, Qt::ConnectionType) [with Func1 = void (QFutureWatcherBase::*)(int); Func2 = MainWindow::on_ButtonRunMulti_clicked()::<lambda()>; typename std::enable_if<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QFutureWatcherBase]', declared using local type 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>', is used but never defined [-fpermissive]

              Evgenii Legotckoi
              • 18. Dezember 2017 08:13

              А вообще, можно и сигнал/слотовое соединение переписать.

              connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](int progressValue){ui->ProgressBar2->setValue(progressValue);});
              Сигнал же уже в качестве аргумента передаёт изменение прогресса.
                М
                • 18. Dezember 2017 08:22

                при такой записи прогресс бар остается на месте, qDebug ничего не пишет. видимо значение  progressValue не определенно. думаю, нужна ссылка на значение прогресса


                  Evgenii Legotckoi
                  • 18. Dezember 2017 08:28
                  • (bearbeitet)

                  Я конечно сталкивался с тем, что некоторый функционал Qt не работает под некоторыми ОС, но то, что не передаётся информация о прогрессе из QFuture - это сомнительно. А вы устанавливаете диапазон значений для прогресс бара?

                  connect(&futureWatcher, &QFutureWatcher<void>::progressRangeChanged, ui->ProgressBar2, &QProgressBar::setRange);
                  Возможно из-за того, что значения диапазона изначально не определены, то и значения текущего прогресса, передаваемые из QFuture, являются ошибочными.
                    М
                    • 18. Dezember 2017 08:35

                    оно у меня стоит от 0 до 100

                    сделал так, все еще не работает
                    connect (& futureWatcher, & QFutureWatcher <void> :: progressRangeChanged,  ui->ProgressBar2, & QProgressBar::setRange);
                    connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged,  ui->ProgressBar2, & QProgressBar :: setValue)
                      Evgenii Legotckoi
                      • 18. Dezember 2017 08:36

                      Что касается этой ошибки, то по факту здесь запрещено копирование QFuture объекта, которое происходит в данном коде.
                      Можно переписать так для данной лямбды.

                      connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, 
                      [this, &futureWatcher](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});
                        М
                        • 18. Dezember 2017 08:38

                        может быть мое окно не активно, пока работает

                        futureWatcher.setFuture(QtConcurrent::map(Perebor,[=](const double& d){ StrategyCod(d,this); }));
                          М
                          • 18. Dezember 2017 08:44

                          Все работает если включить dialog.exec(); . Но как сделать так, что бы работало без диалогового окна?

                            Evgenii Legotckoi
                            • 18. Dezember 2017 08:53

                            То есть вы пытаетесь использовать QProgressBar, а также не удалили код из самого начала этого топика, который относится к QProgressDialog?

                            ... Тогда понятно. Скорее всего слот у вас выполняется раньше, чем успевает закончится расчёт. Диалог QProgressDialog нужно удалять отсюда, если он Вам уже не требуется.
                            А вот задержку выполнения кода можно попробовать сделать через QEventLoop
                            QEventLoop loop;
                            loop.connect(&futureWatcher, QFutureWatcher<void>::finished, &loop, &QEventLoop::exit);
                            // Некоторый код для QtConcurrent
                            futureWatcher.setFuture(QtConcurrent::map(Perebor,[=](const double& d){ StrategyCod(d,this);}));
                            loop.exec()
                              М
                              • 18. Dezember 2017 09:06

                              Дописал без

                              loop.connect(&futureWatcher, QFutureWatcher<void>::finished, &loop, &QEventLoop::exit);
                              и заработало
                                Evgenii Legotckoi
                                • 18. Dezember 2017 09:09

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

                                  М
                                  • 18. Dezember 2017 09:27
                                  • (bearbeitet)

                                  Сделал так

                                  connect ( ui->ButtonStopMulti,  &QPushButton::clicked, &futureWatcher, &QFutureWatcher<void>::pause); //пауза процесса
                                  connect ( ui->ButtonRunMulti,  &QPushButton::clicked,  &futureWatcher, &QFutureWatcher<void>::resume);  //продолжение процесса
                                  и все наконец заработало. Большое спасибо.

                                    Evgenii Legotckoi
                                    • 18. Dezember 2017 09:34

                                    futureWatcher является локальной переменной в вашем коде... возможно, что при установке расчёта на паузу завершается выполнение слота, что ведёт к уничтожению объекта futureWatcher со всеми вытекающими результатами, в том числе и уничтожением сигнал-слотового соединения с кнопкой возобновления процесса.

                                    Закиньте вывод qDebug() << "check"; в конце слота и посмотрите, если будет вывод, то значит слот завершается. нужно иначе продумывать логику, чтобы объекты не уничтожались раньше времени.

                                      Kommentare

                                      Nur autorisierte Benutzer können Kommentare posten.
                                      Bitte Anmelden oder Registrieren
                                      Letzte Kommentare
                                      A
                                      ALO1ZE19. Oktober 2024 08:19
                                      Fb3-Dateileser auf Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                      ИМ
                                      Игорь Максимов5. Oktober 2024 07:51
                                      Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                      d
                                      dblas55. Juli 2024 11:02
                                      QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                      k
                                      kmssr8. Februar 2024 18:43
                                      Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                                      Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                                      Jetzt im Forum diskutieren
                                      J
                                      JacobFib17. Oktober 2024 03:27
                                      добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
                                      JW
                                      Jhon Wick1. Oktober 2024 15:52
                                      Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
                                      КГ
                                      Кирилл Гусарев27. September 2024 09:09
                                      Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
                                      F
                                      Fynjy22. Juli 2024 04:15
                                      при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

                                      Folgen Sie uns in sozialen Netzwerken