- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我将使用 QNetworkAccessManager 通过我的移动应用向服务器发出对 HTTP 服务器的请求。问题是,如何将自定义数据链接到每个请求?我尝试子类化 QNetworkReply 但我发现我必须实现虚拟方法 close()
和 isSequential()
但我不知道它们应该返回什么所以我是恐怕我会破坏网络请求功能。
例如,当我的应用执行登录程序时,它必须存储帐户的电子邮件地址:
class MyApp : public QObject
{
Q_OBJECT
private:
QNetworkRequest request;
QNetworkReply *reply;
QNetworkAccessManager *manager;
...
}
void MyApp::do_log_in(QString email, QString password) {
QString s;
someobject.email=email; // <-- I have to store email address before sending request to server, but where do I store it?
s.append("http://myapp.com/do-auth.php?email=");
s.append(QUrl::toPercentEncoding(email));
s.append("&password=");
s.append(QUrl::toPercentEncoding(password));
connect(manager,SIGNAL(finished(QNetworkReply*)),this,SLOT(login_finished(QNetworkReply*)));
request.setUrl(QUrl(s));
manager->get(request);
}
void MyApp::login_finished(QNetworkReply *rep) {
DepservReply *reply;
QString email;
....
email= ...... // <-- I need to get the email address from QNetworkReply object somehow
///my code here handling server reply
....
}
那么,在我的案例中,我该如何实现电子邮件的存储和检索,我应该子类化哪些类以及我应该重新实现哪些方法?
最佳答案
您可以利用每个 QObject
中可用的动态属性系统并将数据保存在回复中:
// https://github.com/KubaO/stackoverflown/tree/master/questions/network-reply-tracking-40707025
#include <QtNetwork>
class MyCtl : public QObject
{
Q_OBJECT
QNetworkAccessManager manager{this};
// ...
void reply_finished(QNetworkReply *reply);
public:
MyCtl(QObject *parent = nullptr);
void do_log_in(const QString &email, const QString &password);
};
static const char kAuthGetSalt[] = "req_auth-get-salt";
static const char kDoAuth[] = "req_do-auth";
static const char kEmail[] = "req_email";
static const char kPassword[] = "req_password";
static const auto authGetSaltUrl = QStringLiteral("https://myapp.com/auth-get-salt.php?email=%1");
static const auto doAuthUrl = QStringLiteral("https://myapp.com/do-auth.php?email=%1&passwordHash=%2");
MyCtl::MyCtl(QObject *parent) : QObject{parent}
{
connect(&manager, &QNetworkAccessManager::finished, this, &MyCtl::reply_finished);
}
void MyCtl::do_log_in(const QString &email, const QString &password) {
auto url = authGetSaltUrl.arg(email);
auto reply = manager.get(QNetworkRequest{url});
reply->setProperty(kAuthGetSalt, true);
reply->setProperty(kEmail, email);
reply->setProperty(kPassword, password);
}
void MyCtl::reply_finished(QNetworkReply *reply) {
if (!reply->property(kAuthGetSalt).isNull()) {
reply->deleteLater(); // let's not leak the reply
if (reply->error() == QNetworkReply::NoError) {
auto salt = reply->readAll();
auto email = reply->property(kEmail).toString();
auto password = reply->property(kPassword).toString();
Q_ASSERT(!password.isEmpty() && !email.isEmpty());
QCryptographicHash hasher{QCryptographicHash::Sha1};
hasher.addData(salt); // the server must hash the same way
hasher.addData("----");
hasher.addData(password.toUtf8());
auto hash = hasher.result().toBase64(QByteArray::Base64UrlEncoding);
auto url = doAuthUrl.arg(email).arg(QString::fromLatin1(hash));
auto reply = manager.get(QNetworkRequest{url});
reply->setProperty(kDoAuth, true);
reply->setProperty(kEmail, email);
}
}
else if (!reply->property(kDoAuth).isNull()) {
if (reply->error() == QNetworkReply::NoError) {
auto email = reply->property(kEmail).toString();
// ...
}
}
}
通过让编译器检查您使用的标识符是否有效,将常量用作属性名称以避免拼写错误。
上面的示例纠正了代码中的以下关键安全问题:
通过清晰的连接发送安全凭证:使用 https://
,而不是 http://
。
以明文形式发送密码:相反,发送密码的加盐哈希值。创建帐户时,您的服务器应该为每个帐户生成一个随机盐。现有帐户可以不加盐,但一旦用户更改密码,它们就应该加盐。
另请注意,QString
到 QUrl
的转换将自动对字符串进行百分号编码,因此无需显式执行此操作。
关于c++ - 如何让 QNetworkReply 返回自定义数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40707025/
我阅读了信号完成()的文档,它没有说完成()总是发出。我读取了 error() 信号: void QNetworkReply::error(QNetworkReply::NetworkError co
我正在使用 Qt 4.6.3 和以下不起作用的代码 QStringList userInfo; QNetworkRequest netRequest(QUrl("http://api.stackove
我希望能够将 QNetworkReply 保存到 QString/QByteArray。在我看到的示例中,它们总是将流保存到另一个文件。 目前我的代码看起来像这样,我从主机那里得到一个字符串,我想做的
我是 C++ 和 Qt 的新手,所以我很自然地遇到了麻烦。我正在尝试创建一个独立于连接到 URL 并接收 JSON 响应的 MainWindow 的类。但是,使用QObject::connect后,程
我有一个 QWebView我正在通过连接观看网络请求: QObject::connect(page()->networkAccessManager(),
我正在用 Qt 编写一个类,用于从 Bricklink API(一个乐高数据库)中检索信息。它使用 QOAuth1 类进行身份验证,并使用 QNetworkReply 捕获来自 HTTP GET 请求
假设我已经执行了一个 QNetworkRequest 并获得了适当的 QNetworkReply。如果它是一个大文件(比如 1 GB),我如何创建一个 4k 字节数组缓冲区并将数据 4k x 4k 读
我有一个与 QT 相关的问题。 QNetworkAccessManager::get() 返回 QNetworkReply 指针。然后我可以连接到它的完成插槽: QNetworkReply* r =
我可以在 Qt 中使用 QNetworkReply、QNetworkAccessManager、QNetworkRequest 等将文件发布到服务器。该服务器转换文件并将其吐回。我如何获得那个吐出来的
我的应用程序使用 QNetworkReply 从 RESTful API 发送和接收数据。 有许多教程可用于使用 QNetworkReply与 QNetworkAccessManager 一旦找到我使
这个问题在这里已经有了答案: Member access into incomplete type error (3 个答案) 关闭 4 年前。 我正在通过 QNetwork 访问管理器发出一个简单
我将使用 QNetworkAccessManager 通过我的移动应用向服务器发出对 HTTP 服务器的请求。问题是,如何将自定义数据链接到每个请求?我尝试子类化 QNetworkReply 但我发现
我正在尝试通过网络服务捕获一些数据,但遇到了以下问题。通过获取包含特殊字符(“Español”)的字符串,获取的文本不正确(“Espa\u00f1ol”)。 我尝试使用 wireshark 验证数据到
我正在使用 OpenEmbedded 项目的 Qt5 Embedded 层编写嵌入式 RESTful API 客户端。我希望我的客户端能够发送一个简单的异步请求来通知我的服务器正在发生的事情,另一种情
我正在尝试使用 QNetworkManager 上传一些数据并在 QNetworkReply 中获得响应。然后我测试,如果回复是一个错误,如果,我打印错误消息,我不明白结果。 if(reply->er
是一种HTTP请求发送方式。当目标网站响应时,将调用 httpFinished()。 void HTTPClientBase:: HttpRequestGet() { QNetworkRequ
我尝试在 C++ 中对 url https://... 执行 post 方法,但我收到连接关闭错误。 如果我使用另一个 url,比如 https://www.google.gr,我发现我的代码可以工作
是一种HTTP请求发送方式。当目标网站响应时,将调用 httpFinished()。 void HTTPClientBase:: HttpRequestGet() { network_mana
我有以下代码: connect(&netMgr, SIGNAL(finished(QNetworkReply*)), this, SLOT(loadFinished(QNetworkReply*)),
如何在 QWebPage 之前从特定 URL 的 QNetworkReply 响应中读取数据?但是当finished()发出信号 QWebPage 已读取回复,因此连接 readyRead()或调用
我是一名优秀的程序员,十分优秀!