0

我对信号槽机制有疑问,可能是QSharedPointer因为智能指针对我来说是新的。

上下文的一点解释: TCPAssociationLANAssociation. TCPClient是子类LANClientLANClient有受保护的QSharedPointer<LANAssociation> connection变量,TCPClient有私有的QSharedPointer<TCPAssociation> conn;两者都指向在覆盖的纯虚函数中创建的同一个对象,如下所示:

void TCPClient::startNewConnection(const QHostAddress &address)
{
    conn.clear();
    connection.clear();

    conn = QSharedPointer<TCPAssociation>(new TCPAssociation(address, port, QString("Client %1").arg(hostId())));
    Q_ASSERT(conn);

    Q_ASSERT(connect(conn.data(), &LANAssociation::started, this, &LANClient::assoc_started, Qt::UniqueConnection));

    connection = conn.staticCast<LANAssociation>();
    Q_ASSERT(connection);
    conn->start();
    //TCPAssociation::start() { doStuff(); emit started(); }
}

该代码在调试模式下的 Windows 10 上运行良好。在 Linux 的发布模式下,LANClient::assoc_started从不调用。

现在我在 Windows 上运行我的服务器应用程序(上面的函数来自客户端应用程序),我可以看到另一个插槽没有在发布中被调用,但在调试中工作得很好:

void ThreadController::init(ThreadWorker *worker)
{
    worker->moveToThread(&workerThread);

    Q_ASSERT(connect(&workerThread, &QThread::finished, worker, &QObject::deleteLater, Qt::UniqueConnection));
    Q_ASSERT(connect(this, &ThreadController::operate, worker, &ThreadWorker::doWork, Qt::UniqueConnection));
    Q_ASSERT(connect(worker, &ThreadWorker::finished, this, &ThreadController::finished, Qt::BlockingQueuedConnection));
    doConnects(worker);

    workerThread.start();
}

*worker指向 的子类ThreadWorkerThreadWorker::doWork()是虚函数。它是在调试中调用的,而不是在发布时调用的。我通过制作doWork()纯虚拟来修复它 - 现在它在调试和发布中都被调用。

我也有问题,QTcpSocket对象QUdpSocket不发射connected()or readyRead()。现在我明白这可能是因为我QSharedPointer<QAbstractSocket>用来管理它们,但是:

  1. 为什么调试和发布之间的区别?如何正确地做到这一点?

  2. 我应该停止使用继承还是共享指针?

  3. 为什么TCPAssociation::started()不被发射?LANClient::assoc_started也不是虚函数。
4

1 回答 1

1

宏内部的表达式Q_ASSERT将不会在非调试构建配置中进行评估:

Q_ASSERT()对于在开发过程中测试前置条件和后置条件很有用。QT_NO_DEBUG如果在编译期间定义,它什么也不做。

https://doc.qt.io/qt-5/qtglobal.html#Q_ASSERT

于 2020-02-28T08:34:48.597 回答