Qt 使用 QNetworkReply 获取外部 IP 地址

Posted

技术标签:

【中文标题】Qt 使用 QNetworkReply 获取外部 IP 地址【英文标题】:Qt get external IP address using QNetworkReply 【发布时间】:2017-12-16 10:00:50 【问题描述】:

美好的一天

简介:

我的应用程序需要获取外部 IP 地址并将其与内部获取的地址匹配,从而允许应用程序继续进行。

为此,我使用了QNetworkAccessManager 和QNetworkReply。

我的代码是使用this 示例作为参考构建的。

我尝试过的:

可以通过从ipify API获取一个JSON对象来获取外部IP。

我通过以下方式确认了这一点:

curl "https://api.ipify.org?format=json"

它反过来以我当前的 IP 地址响应,格式为:

"ip":"255.255.255.255"

这是一个JSonObject。使用它,我创建了下面的代码。

问题:

问题很简单,我没有得到任何回应。 post 请求已执行,但根本没有触发响应(或 finished)信号。

POST -> GET 请求

我已更改get 请求的代码,因为这解决了this thread 上的无响应问题。

我通过在 URL 中使用查询参数指定整个 url 来做到这一点:

QNetworkRequest request(QUrl("https://api.ipify.org?format=json"));

包括标头内容类型和大小(如下例,最后调用QNetworkAccessManager::get()

replyExternalAddress = networkManager->get(request);

但这也没有回应。

我认为这是我缺少的一些小东西,但我根本看不到它。

建议?


查询外部IP代码:

// public callable method, starting network request
void APICommunicator::requestExternalAddress()
    qInfo(apicommunicator) << "Requesting external IP address from ipify.org";

    // creates network request
    // specifies "format=json"
    QUrlQuery postData;
    postData.addQueryItem("format", "json");
    QByteArray encodedQuery = postData.toString(QUrl::FullyEncoded).toUtf8();
    QNetworkRequest request(QUrl("https://api.ipify.org"));

    request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
    request.setHeader(QNetworkRequest::ContentLengthHeader, QString::number(encodedQuery.size()));

    // creates merged URL from URL and query items and sends a post:
    //    https://api.ipify.org?format=json
    replyExternalAddress = networkManager->post(request, encodedQuery);

    // Creates QMetaObject::Connection connection for finished signal from QNetworkReply
    conExternalAddress = QObject::connect(replyExternalAddress, SIGNAL(finished()), this, SLOT(externalAddressResponse()));

    // attach error listener to reply
    addErrorListener(replyExternalAddress, conExternalAddress);




void APICommunicator::externalAddressResponse()
    qDebug(apicommunicator) << "External Address response recieved";

    // disconnect signals
    QObject::disconnect(conExternalAddress);
    QObject::disconnect(conErrorListener);

    // read all output from JSon object
    QByteArray ba = replyExternalAddress->readAll();

    // delete QNetworkReply
    replyExternalAddress->deleteLater();

    LogMessageHandler::writeToApiLog(QString("\n\nCALL EXTERNAL [" + replyExternalAddress->request().url().toString() + "]\n" + QString(ba)));

    QJsonObject doc = QJsonDocument::fromJson(ba).object();
    QString ip = doc.value("ip").toString();    
    QHostAddress address = QHostAddress();

    if (ip.isEmpty()) 
        qWarning(apicommunicator) << "External Address: no data received";
    
    else 
        address = QHostAddress(version);
    

    // replies with address to external slot (in main application)
    emit ExternalAddressReply(address);

【问题讨论】:

【参考方案1】:

问题是您发送的是POST 请求,而ipify.org 只需要GET 请求。您似乎误以为您需要发送POST 请求才能随请求一起发送参数(format=json),这是不正确的。在您的代码中,您将参数作为 POST 数据发送,这与您在浏览器中提交 Web 表单时使用的技术相同(因为您将内容类型标头设置为 application/x-www-form-urlencoded)。

您绝对不需要模仿 Web 浏览器发送表单的请求,以便能够与 ipify.org 提供的 API 对话。 ipify.org 提供了一个更简单的界面;您只需要在获取请求中发送您的查询字符串。 Qt 通过提供类QUrlQuery 使工作变得更加容易,该类提供了一种构建url 查询的方法。这是一个工作示例:

#include <QtCore>
#include <QtNetwork>

int main(int argc, char *argv[]) 
    QCoreApplication a(argc, argv);

    QNetworkAccessManager networkManager;

    QUrl url("https://api.ipify.org");
    //the query used to add the parameter "format=json" to the request
    QUrlQuery query;
    query.addQueryItem("format", "json");
    //set the query on the url
    url.setQuery(query);

    //make a *get* request using the above url
    QNetworkReply* reply = networkManager.get(QNetworkRequest(url));

    QObject::connect(reply, &QNetworkReply::finished,
                     [&]()
        if(reply->error() != QNetworkReply::NoError) 
            //failure
            qDebug() << "error: " << reply->error();
         else  //success
            //parse the json reply to extract the IP address
            QJsonObject jsonObject= QJsonDocument::fromJson(reply->readAll()).object();
            QHostAddress ip(jsonObject["ip"].toString());
            //do whatever you want with the ip
            qDebug() << "external ip: " << ip;
        
        //delete reply later to prevent memory leak
        reply->deleteLater();
        a.quit();
    );
    return a.exec(); 

【讨论】:

我尝试了 get,但是我创建了一个完整的 URL,其中包含 URL 中的查询。为了利益,为什么这行不通? @KGCybeX,嗯,像this 这样的东西非常适合我。我认为您遇到的问题与代码中的其他内容有关。 无论如何,感谢您为回答问题所做的努力。我将应用解决方案并提供反馈 不客气。无论如何,在更改您问题中的代码以发送GET 请求时,我无法重现您的问题。我认为问题出在您代码中的其他地方...

以上是关于Qt 使用 QNetworkReply 获取外部 IP 地址的主要内容,如果未能解决你的问题,请参考以下文章

Qt - 发出请求后QNetworkReply为空[重复]

Qt QNetworkReply readAll 在再次请求相同的 url 时返回空

如何用QT5做一个获取网页源码的东西

Qt5 获取天气数据时报这个错误是啥意思

QnetworkReply 错误 - Symbian Qt

Qt Jambi:访问 QNetworkReply 的内容