c++ - QThread中的QTcpSocket将commitTransaction但是当调用Write时 "Cannot create children for a parent that is in a different thread."

标签 c++ qt qthread qtcpsocket qtcpserver

免责声明:我对 Qt 和围绕线程和网络的任何类型的编程都比较陌生。我还采用了 Qt 示例、API 和其他在线示例中的大量代码。

所有代码都可以找到on GitHub .这段代码相对简单,因为它可以去掉 GUI。我认为以这种方式提供它与仅粘贴下面的代码相比也会有所帮助。

我想使用并且相信我需要使用线程,因为我需要多个客户端向服务器发送请求,服务器运行一些 SQL 代码,然后将结果吐回客户端(基本上派生一个 MySQL 服务器,但是具体到我在做什么)。不过现在,我只是在努力学习这一切的运作方式。

综上所述,如标题所述。我的客户端可以连接到服务器,服务器设置线程,并将通过 readReady 接收数据(字符串)。读入数据后,现在我只是试图将其回显给客户端。它会这样做,但只有一次。然后它吐出:

QObject: Cannot create children for a parent that is in a different thread.
(Parent is QNativeSocketEngine(0x266cca92ea0), parent's thread is serverThread(0x266cca9ed60), current thread is QThread(0x266cac772e0)

我无法向服务器发送任何进一步的数据,除非我让客户端重新连接,然后在发送数据后,它会完成它的工作,但随后会吐出同样的错误并停止运行。我尝试了很多不同的东西,但似乎无法解决问题。我什至尝试按照 API 中的建议为此设置一个 SIGNAL/SLOT:

It is important to remember that a QThread instance lives in the old thread that instantiated it, not in the new thread that calls run(). This means that all of QThread's queued slots will execute in the old thread. Thus, a developer who wishes to invoke slots in the new thread must use the worker-object approach; new slots should not be implemented directly into a subclassed QThread.

无论如何,我们将不胜感激!我的代码在下面..

服务器

服务器线程.cpp

// Project
#include "ServerDialog.h"
#include "ServerThread.h"

ServerThread::ServerThread(qintptr _socketDiscriptor, QObject *parent /*= 0*/)
    : QThread(parent)
{
    socketDiscriptor = _socketDiscriptor;
}

void ServerThread::run()
{
    emit threadStarted(socketDiscriptor);

    // Start Thread
    clientSocket = new QTcpSocket;

    // Set SocketDisc
    if (!clientSocket->setSocketDescriptor(socketDiscriptor))
    {
        emit error(clientSocket->error());
        return;
    }
    
    // Connect Socket and Signal
    connect(clientSocket, SIGNAL(readyRead()), this, SLOT(readyRead()));
    connect(clientSocket, SIGNAL(disconnected()), this, SLOT(disconnected()));

    //// Loop Thread to Stay Alive for Signals and Slots
    exec();
}

void ServerThread::readyRead()
{
    QDataStream in(clientSocket);
    in.setVersion(QDataStream::Qt_5_7);
    in.startTransaction();
    QString dataReceived;
    in >> dataReceived;
    if (!in.commitTransaction())
    {
        emit readyReadError(socketDiscriptor);
        return;
    }
    emit readyReadMessage(socketDiscriptor, dataReceived);
    echoData(dataReceived);
}

void ServerThread::disconnected()
{
    emit threadStopped(socketDiscriptor);
    clientSocket->disconnect();
    clientSocket->deleteLater();
    this->exit(0);
}

void ServerThread::echoData(QString &data)
{
    QByteArray block;
    QDataStream out(&block, QIODevice::WriteOnly);
    out.setVersion(QDataStream::Qt_5_7);
    out << data;
    clientSocket->write(block);
}

因此在 ServerThread.cpp 中调用 echoData 时,即出现错误且 Socket 停止运行时。

我们将不胜感激任何帮助。我知道还有一些关于线程的“无法为...创建子项”的其他帖子。但我没有发现它们有任何帮助。我确实觉得有趣但不明白的一件事可能是使用 moveToThread(),但对此有很多不同的评论。

与仅通过解释或 API 指针相比,我通过代码示例和解释学得最好。谢谢!

最佳答案

大部分Qt网络函数都是异步的;他们不会阻塞调用线程。 如果您使用的是QTcpSocket,则无需搞乱线程。事实上,为每个套接字创建一个线程是一种矫枉过正,因为该线程将把大部分时间花在等待某些网络操作完成上。下面是我将如何在 Qt 中实现单线程回显服务器:

#include <QtNetwork>
#include <QtCore>

//separate class for the protocol's implementation
class EchoSocket : public QTcpSocket{
    Q_OBJECT
public:
    explicit EchoSocket(QObject* parent=nullptr):QTcpSocket(parent){
        connect(this, &EchoSocket::readyRead, this, &EchoSocket::EchoBack);
        connect(this, &EchoSocket::disconnected, this, &EchoSocket::deleteLater);
    }
    ~EchoSocket() = default;

    Q_SLOT void EchoBack(){
        QByteArray receivedByteArray= readAll();
        write(receivedByteArray);
        disconnectFromHost();
    }
};

class EchoServer : public QTcpServer{
public:
    explicit EchoServer(QObject* parent= nullptr):QTcpServer(parent){}
    ~EchoServer() = default;

    //override incomingConnection() and nextPendingConnection()
    //to make them deal with EchoSockets instead of QTcpSockets
    void incomingConnection(qintptr socketDescriptor){
        EchoSocket* socket= new EchoSocket(this);
        socket->setSocketDescriptor(socketDescriptor);
        addPendingConnection(qobject_cast<QTcpSocket*>(socket));
    }

    EchoSocket* nextPendingConnection(){
        QTcpSocket* ts= QTcpServer::nextPendingConnection();
        return qobject_cast<EchoSocket*>(ts);
    }
};

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

    EchoServer echoServer;
    echoServer.listen(QHostAddress::Any, 9999);

    QObject::connect(&echoServer, &EchoServer::newConnection, [&](){
        EchoSocket* socket= echoServer.nextPendingConnection();
        qDebug() << "Got new connection from: " << socket->peerAddress().toString();
    });

    return a.exec();
}

#include "main.moc"

注意事项:

  • 此服务器有能力同时处理多个客户端,因为没有阻塞。线程将只响应以适当的操作发生的事件;因此,如果该事件是一个新连接,它将创建一个新的 EchoSocket 对象来处理它并向 qDebug() 打印一条语句,如果该事件正在接收先前创建的套接字上的某些东西,同一个线程将回显接收到的数据并关闭连接。 它永远不会阻塞在等待数据到达的单个连接上,也不会阻塞等待新连接到达
  • 因为您提到在项目的后期使用一些 SQL 查询来响应某些连接。请避免线程化,因为 Qt 中的 SQL 数据库连接只能从创建它的线程使用,参见 docs here .因此,您将不得不为应用程序中的每个线程(并因此为每个连接)创建一个新的数据库连接(这不仅仅是矫枉过正),或者稍后切换到单线程设计。

在本节中,我将解释为什么线程不适合您的工作方式:

您应该在您的QThread 子类中声明插槽,而是使用worker QObject并将它们移动到QThread根据需要

您在问题中提供的引述是您收到此警告的确切解释。您创建的 ServerThread 实例将存在于主线程(或创建它的任何线程)中。现在让我们从您的代码中考虑这一行:

connect(clientSocket, SIGNAL(readyRead()), this, SLOT(readyRead()));

信号 readyRead()从当前 ServerThread 实例发出(因为 clientSocket 对象发射它存在于那里),然而,接收者对象是当前的 ServerThread 实例,但它存在于 ma​​in thread 中。这是documentation says :

If the receiver lives in the thread that emits the signal, Qt::DirectConnection is used. Otherwise, Qt::QueuedConnection is used.

现在,Qt::QueuedConnection 的要点是在接收器对象的线程中执行插槽。这意味着,您的插槽 ServerThread::readyRead()ServerThread::disconnected 将在主线程中执行。这很可能不是您想要的,因为您最终将从主线程访问 clientSocket。之后,任何导致创建子 QObjectclientSocket 调用都会导致您收到警告(您可以看到 QTcpSocket::write() 做这个 here )。

关于c++ - QThread中的QTcpSocket将commitTransaction但是当调用Write时 "Cannot create children for a parent that is in a different thread.",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41083764/

相关文章:

c++ - Windows (Visual C) 是否有 unistd.h 的替代品?

qt - 无法将 [undefined] 分配给 QColor

c++ - 在不同的无限循环上同步 Qthreads (QTimerEvent)

c++ - 在 GUI 线程之外打印 QWidget(渲染)

python - 如何将值从一个对话框传递到另一个对话框

QThread 中的 Qt 信号和槽

c++ - 将固定长度的记录写入文件 C++

c++ - 动态分配大于SIZE_T/UINT的内存空间(即在堆上)

c++ - 在 gluSphere 上绘制的 OpenGL 线?

c++ - Qt检查当前进程是32位还是64位