QT错误中的C ++线程亲和力

Posted

技术标签:

【中文标题】QT错误中的C ++线程亲和力【英文标题】:C++ thread affinity in QT error 【发布时间】:2013-08-26 20:27:26 【问题描述】:

我有一个简单的 Qt 应用程序,可以将文件从文件夹发送到服务器。

当没有互联网连接时,预期的行为是将文件保存在“离线”文件夹中以备后用。然后,几分钟后,应用程序本身(在不同的线程中)将不得不检查该文件夹以查看是否有一些文件正在等待发送。

这是一些代码,我去掉了一些不必要的行:

main.cpp:

int main(int argc, char *argv[])

    QApplication app(argc, argv);

    MainWin win;
    win.show();

    return app.exec();

mainwin.cpp:

MainWin::MainWin(QWidget * parent) : QWebView(parent)


    m_network = new QNetworkAccessManager(this);

    m_communicationHandler = new TestNetwork(this, m_network);

    // harvie
    Harvester * harvie = new Harvester(m_network);
    harvie->start();



harvester.cpp:

标题:

class Harvester : public QThread

    Q_OBJECT
public:
    QNetworkAccessManager * m_network;
    Harvester(QNetworkAccessManager * m_network);
...

来源:

Harvester::Harvester(QNetworkAccessManager * m)

    qDebug() << "harvie start" << endl;
    this->m_network = m;

...

bool Harvester::sendFile(QFileInfo *fileInfo) 

    connect(this->m_network,SIGNAL(finished(QNetworkReply*)),this,SLOT(validateReply(QNetworkReply*)));

    QHttpMultiPart *multiPart = new QHttpMultiPart(QHttpMultiPart::FormDataType);

    QHttpPart filePart;
    filePart.setHeader(QNetworkRequest::ContentTypeHeader, QVariant("text/xml")); // @todo test
    filePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"someFile\"; filename=\"" + fileInfo->baseName() + ".xml\""));

    QFile *file = new QFile(fileInfo->filePath());
    file->open(QIODevice::ReadOnly);
    filePart.setBodyDevice(file);
    file->setParent(multiPart); // we cannot delete the file now, so delete it with the multiPart


    multiPart->append(filePart);

    qDebug() << "post" << endl;

    QNetworkReply *reply = this->m_network->post(QNetworkRequest(QUrl("XXX")), multiPart);

    multiPart->setParent(reply); // delete the multiPart with the reply

    return true;

这是我收到的错误:

Warning: QObject: Cannot create children for a parent that is in a different thread.
(Parent is QNetworkAccessManager(0x3e46c30), parent's thread is QThread(0x3dd6d50), current thread is Harvester(0x3e47eb8)

我猜 QNetworkAccessManager 在尝试从另一个线程发送请求时会发生冲突,即使我使用相同的指针也是如此。

我还没有尝试过的另一个解决方案是将harvester 作为一个单独的进程运行,但首先我想通过线程来解决它。

谢谢

【问题讨论】:

【参考方案1】:

qt thread with movetothread

http://crpppc19.epfl.ch/doc/qt4-doc-html/html/qthread.html#details

http://crpppc19.epfl.ch/doc/qt4-doc-html/html/qobject.html#moveToThread

试试这个:

Harvester::Harvester(QNetworkAccessManager * m)

    qDebug() << "harvie start" << endl;
    this->m_network = m;

    m_network->moveToThread(this); // Change thread affinity!

希望对您有所帮助。

【讨论】:

我在答案中发布的第一个链接显示了一种设置工作线程的方法。您可以将 QNetworkAccessManager 的所有初始化移动到工作人员/任务中。只需确保将其移至正确的线程即可。 我已经以 4 或 5 种不同的方式处理 QThreads。此博客:mayaposch.wordpress.com/2011/11/01/… 促使 QThread 文档中的一些更改以更好地适应“工人”/“生产者”模型。您可以将 QThread 子类化,但您需要注意如何清理所有内容、如何连接所有内容以及初始化线程在 run() 函数顶部使用的 qobjects。 好的,谢谢你的帮助,我明天试试,如果我成功了,会告诉你的。 很高兴我能帮上忙。过去,我对线程亲和性垃圾感到很头疼。 Qt 开发者日涵盖了一系列很棒的主题,包括多线程:qt-project.org/videos#c-89

以上是关于QT错误中的C ++线程亲和力的主要内容,如果未能解决你的问题,请参考以下文章

VS 2010 中的 C++ 调试访问冲突

Qt Creator 2.7.0 (Qt 5.0.2) 中的 C++11 线程支持

C中多线程python扩展中的段错误

多线程导致Qt中的运行时错误

从 Qt 中的错误中恢复

循环C ++中的分段错误Openmp