Qt C++ как правильно удалить QFutureWatcher?
Мой код иногда дает сбой из-за асинхронности будущего.
Это не производственный код, а просто очень простой материал для демонстрации того, что я хочу делать.
Мне нужно загрузить некоторую информацию, а затем вернуть первый полученный результат. Только представьте, я не хочу ждать, поэтому я используюQFutureWatcher
слишком ускорить загрузку.
Когда я получил результат (first
или второй instance
) Я хочу вернуть результат НЕМЕДЛЕННО (то есть я не хочу использовать QFutureWatcher::waitForFinished
. (Если я получил сигнал завершения отfirst
Например, мне больше не нужен second
один. Или то же самое дляfirst
.
В производственном коде функция download
я не могу получить доступ к QNetworkReply
, поэтому ожидающий запрос не может быть прерван.
Иногда этот код дает сбой. Что я делаю не так?
#include <QFutureWatcher>
#include <QtConcurrent>
#include <QEventLoop>
#include <QTimer>
#include <QNetworkReply>
#include <QNetworkRequest>
#include <QNetworkAccessManager>
QByteArray download(const QUrl &url)
{
QNetworkAccessManager manager;
QNetworkRequest req(url);
req.setAttribute(QNetworkRequest::SynchronousRequestAttribute, true);
QNetworkReply *reply = manager.get(req);
reply->deleteLater();
return reply->readAll();
}
QByteArray downloadSmth(const QUrl &urlFirst, const QUrl &urlSecond)
{
typedef QFutureWatcher<QByteArray> FutureWatcher;
FutureWatcher *first = new FutureWatcher;
FutureWatcher *second = new FutureWatcher;
QByteArray res;
QEventLoop loop;
QObject::connect(first, &FutureWatcher::finished, [&res, &loop, first] {
res = first->result();
loop.quit();
});
QObject::connect(second, &FutureWatcher::finished, [&res, &loop, second] {
res = second->result();
loop.quit();
});
QTimer timer;
timer.setSingleShot(true);
QObject::connect(&timer, &QTimer::timeout, &loop, &QEventLoop::quit);
first->setFuture(
QtConcurrent::run([&urlFirst] {
return download(urlFirst);
}));
second->setFuture(
QtConcurrent::run([&urlSecond] {
return download(urlSecond);
}));
timer.start(60 * 1000);
loop.exec();
if (timer.isActive()) {
timer.stop();
}
first->deleteLater();
second->deleteLater();
return res;
}