我在Qt控制台应用程序中有以下两个简单方法:
void Webservice::getFile(PostElement el)
{
parameter = ⪙
QUrl url(PATH);
QUrlQuery query;
query.addQueryItem(el.getParam(), el.getValue());
url.setQuery(query);
request.setUrl(url);
manager->get(request);
connect(manager, SIGNAL(finished(QNetworkReply*)), this,
SLOT(downloadCompleted(QNetworkReply*)));
}
void Webservice::downloadCompleted(QNetworkReply *reply)
{
QByteArray b = reply->readAll();
qDebug() << "Download completed!";
QFile file("/tmp/" + parameter->getValue());
file.open(QIODevice::WriteOnly);
file.write(b);
file.close();
reply->deleteLater();
}
我创建了一个QTest项目来测试getFile
方法。
#include <QtTest>
#include <QCoreApplication>
// add necessary includes here
#include "webservice.h"
class WebserviceTest : public QObject
{
Q_OBJECT
public:
WebserviceTest();
~WebserviceTest();
private:
Webservice ws;
private slots:
void initTestCase();
void cleanupTestCase();
void getFile();
};
WebserviceTest::WebserviceTest()
{
}
WebserviceTest::~WebserviceTest()
{
}
void WebserviceTest::initTestCase()
{
}
void WebserviceTest::cleanupTestCase()
{
}
void WebserviceTest::getFile()
{
PostElement el{"file", "myPic.png"};
ws.getFile(el); // it should wait for the slot to be executed
}
QTEST_MAIN(WebserviceTest)
#include "tst_webservicetest.moc"
如您所见,它有信号和插槽。如果我在原始项目中运行那个,我看到downloadCompleted
执行没有问题(因为它使用事件循环)。但是在测试项目中,不会调用插槽。我用谷歌搜索并找到了一些使用QSignalSpy的例子,但目前我没有成功执行插槽。
如何告诉我的测试“等待”插槽完成?
这正是QSignalSpy
的目的,它基本上旋转了等待信号执行的新QEventLoop
。您的测试启动和异步操作,但它不会等待它完成,因此它将在操作正在进行时销毁对象。
Qt测试还有一些内置的超时功能,可以防止在使用QSignalSpy
时停止测试。在您的情况下,您需要从类Webservice
发出新信号,表明下载已完成,例如(如果需要,您也可以将参数传递给信号):
class Webservice : public QObject {
Q_OBJECT
// Constructors etc.
signals:
void finished();
}
然后在你的void Webservice::downloadCompleted(QNetworkReply *reply)
结束时:
emit finished();
在你的测试中这样的事情:
PostElement el{"file", "myPic.png"};
// Prepare spy
QSignalSpy spy(&ws, SIGNAL(finished));
ws.getFile(el);
// Wait for result - by default this waits 5 seconds
spy.wait();
// Check if we had a timeout or actually signal was raised
QCOMPARE(spy.count(), 1);