Ожидая сигнала

Я работаю над приложением, которое загружает содержимое файла на сервер.

Чтобы загрузить файл на сервер, я использую класс QNetworkAccessManager. Поскольку он работает как асинхронный путь, я изменил его, чтобы работать синхронно, используя QEventLoop.

Class FileTransfer
{
Public : 
     QNetworkAccessManager mNetworkManager;
     Void Upload(QNetworkRequest request, QIODevice *data)
     {
           responce = mNetworkManager.put(request, data);
           EventLoop.exec();
           ReadResponce(responce);
      }

      Void Stop()
      {
            responce ->close();
      }
}

В моем примере приложения у меня есть 2 окна. 1-й для выбора файлов и 2-го для отображения прогресса.

Когда пользователь нажимает кнопку загрузки в первом окне, отображается второе окно, а затем я создаю объект FileTransfer и начинаю загрузку.

При загрузке файла, если пользователь закрывает форму, а затем в деструкторе окна, я вызываю остановку "FileTransfer" после этого, я удаляю объект FileTransfer.

Но здесь функция Upload() еще не завершена, поэтому она выйдет из строя.

Пожалуйста, помогите мне: Как подождать в функции "stop()" до тех пор, пока функция Upload() не будет завершена

Ответ 1

Из того, что я вижу из вашего кода, вы выполняете QEventLoop, но на самом деле вы не подключаете свой "quit" слот к любому сигналу. Возьмите ниже, как пример, логин - это QHttp - и код взят из чего-то другого - но принцип применяется.

/* Create the QEventLoop */
QEventLoop pause;
/* connect the QHttp.requestFinished() Signal to the QEventLoop.quit() Slot */
connect(&login, SIGNAL(requestFinished( int, bool )), &pause, SLOT(quit()));
/* The code that will run during the QEventLoop */
login.request(header,&logmein,&result);
/* Execute the QEventLoop - it will quit when the above finished due to the connect() */
pause.exec();

Это может быть применено к вашему коду, если я не ошибаюсь, как это...

/* connect the signal to the relevant slot */
connect(&mNetworkManager, SIGNAL(finished( QNetworkReply )), &EventLoop, SLOT(quit()));
/* Execute the code that will run during QEventLoop */
responce = mNetworkManager.put(request, data);
/* Execute the QEventLoop */
EventLoop.exec();

Извините, если я ошибаюсь в вашем запросе! После перерыва я снова поймаю qt снова, но я считаю, что это то, что вы имеете в виду! Удачи!

Ответ 2

Я думаю, вам нужно добавить что-то подобное в своей функции загрузки:

if (upf->openFile())
{
    reply = manager->post(request, upf);
    connect(reply, SIGNAL(uploadProgress(qint64,qint64)), this, SIGNAL(progress(qint64,qint64)));
    connect(reply, SIGNAL(finished()), this, SLOT(replyFinished()));
    isInProgress = true;
    emit started();
} else
{
    emit finished(true, false, tr("Error: can't open file %1").arg(filename));
}

Вот полный текст: datacod-qt-tools

Надеюсь, что это поможет.

Ответ 3

Лично я бы рекомендовал не использовать какие-либо из этих ответов. Достаточно было бы подключить защелку обратного отсчета к сигналу.

Итак, вы можете написать:

Latch latch( 1 );
QObject::connect( reply, SIGNAL(finished()),
                  &latch, SLOT(countDown()) );

latch.wait();

Для этого вам понадобится обертка:

class Latch : public QObject {
    Q_OBJECT
public:
    Latch( uint count );
    void wait();
public slots:
    void countDown();
private:
    gcl::countdown_latch _latch;
};