• Qt实现同步(阻塞式)http get等网络访问操作


    从Qt4.4开始,引入了QNetworkRequest、QNetworkReply 和 QNetworkAccessManager等类来进行HTTP、FTP的操作,替代之前的QFtp和QHttp。很多情况下采用QNetworkAccessManager的finished信号构建异步方式。

     1 //构建一个manager对象
     2 QNetworkAccessManager *manager = new QNetworkAccessManager(this); 
     3 //manager具有异步API,当http请求完成后,会通过finished信号进行通知
     4 connect(manager,&QNetworkAccessManager::finished,this,&MyClass::replyFinished); 
     5 //发送异步get请求
     6 manager->get(QNetworkRequest(QUrl("http://qt-project.org")));
     7  
     8 //这里也可以用一个QEventLoop来等待请求完成,但是我更爱用槽函数
     9 //QNetworkReply *reply=manager->get(request);
    10 //QEventLoop eventLoop;
    11 //connect(manager, &QNetworkAccessManager::finished, &eventLoop, &QEventLoop::quit);
    12 //eventLoop.exec();
    13 //QByteArray reply_data=reply->readAll();

    Qt的网络操作类是异步(非阻塞的),但有时想做一些阻塞的事情就不方便了,特别是需要在当前函数中直接获得返回值时,则可以使用QEventLoop阻塞运行,上面的注释部分。具体的用例如下代码所示:

     1 QByteArray MyNetworkAccess::get(const QString &strUrl)
     2 {
     3     assert(!strUrl.isEmpty());
     4 
     5     const QUrl url = QUrl::fromUserInput(strUrl);
     6     assert(url.isValid());
     7 
     8     QNetworkRequest qnr(url);
     9     QNetworkReply* reply = m_qnam.get(qnr); //m_qnam是QNetworkAccessManager对象
    10 
    11     QEventLoop eventLoop;
    12     connect(reply, &QNetworkReply::finished, &eventLoop, &QEventLoop::quit);
    13     eventLoop.exec(QEventLoop::ExcludeUserInputEvents);
    14 
    15     QByteArray replyData = reply->readAll();
    16     reply->deleteLater();
    17     reply = nullptr;
    18 
    19     return replyData;
    20 }

    当然如上方式不支持重定向(301等),因为暂时用不上,如果要支持,还要在return前判断并循环或递归。

    另外如果出现error,上述方式会把服务器返回的错误信息直接返回,支持判断错误的版本请继续往下看!

    并且可以看出本来是封装了一个网络操作类,但现在只有get,post等还没做,等弄好了也一起放上来!

    第二版——支持判断error和重定向(error和重定向均按错误处理):

     1 QByteArray MyNetworkAccess::get(const QString &strUrl)
     2 {
     3     assert(!strUrl.isEmpty());
     4 
     5     const QUrl url = QUrl::fromUserInput(strUrl);
     6     assert(url.isValid());
     7 
     8     QNetworkRequest qnr(url);
     9     QNetworkReply* reply = m_qnam.get(qnr); //m_qnam是QNetworkAccessManager对象
    10 
    11     QEventLoop eventLoop;
    12     connect(reply, &QNetworkReply::finished, &eventLoop, &QEventLoop::quit);
    13     eventLoop.exec(QEventLoop::ExcludeUserInputEvents);
    14 
    15     QByteArray replyData = reply->readAll();
    16     int statusCode = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
    17     QVariant redirectAttr = reply->attribute(QNetworkRequest::RedirectionTargetAttribute);
    18     if (reply->error()
    19         || 300 == statusCode //状态码300 Multiple Choices,既不是错误也不算重定向,应该是qt bug
    20         || !redirectAttr.isNull())
    21     {
    22         QString errString = reply->error() ? reply->errorString() : QString("发生重定向(%1),不允许此情况").arg(statusCode);
    23         QMessageBox::critical(nullptr, "网络异常",
    24             QString("发送get请求时出现错误:
    网址:%1
    错误信息:%2").arg(reply->request().url().toDisplayString(), errString));
    25         replyData.clear();
    26     }
    27 
    28     reply->deleteLater();
    29     reply = nullptr;
    30 
    31     return replyData;
    32 }

     注意:以下方式可不行哦

    1、通过QNetworkReply中的isFinished()或isRunning函数,在while循环中判断reply是否已经结束——不可行

    转自:https://www.cnblogs.com/roadbike/p/6055263.html

  • 相关阅读:
    Linux下的输入/输出重定向
    strcpy与strncpy的区别
    C++的函数重载
    gtest 学习二
    gtest 学习一
    char* wchar* char wchar转换
    iconv 编译不通过问题
    嵌入式常用库
    驱动编程class_create说明
    libiconv 交叉编译
  • 原文地址:https://www.cnblogs.com/liushui-sky/p/13814416.html
Copyright © 2020-2023  润新知