3

我真的很难过,希望有人知道我的问题。

我有一个非常简单的 SSL 客户端和服务器。连接很好。沟通很好。当客户端与服务器断开连接时,就会出现问题。这会在服务器上触发一个信号,该信号在 SLOT 中处理error_handler(QAbstractSocket::SocketError in_error)sslSocket我想,在那个函数中,必须删除对象。

但是,这样做会导致服务器出现段错误。我不明白发生了什么事。我希望这非常简单,但显然我错过了一些 Qt(或其他)概念。

有人可以帮忙吗?

基本服务器代码:

void SSLServer::incomingConnection(int sd)
{
    sslSocket = new SSLSocket(this);
    if( sslSocket->setSocketDescriptor(sd))
    {
        QFile sslkeyfile(privKey_);
        sslSocket->setPrivateKey(QSslKey(sslkeyfile.readAll(),QSsl::Rsa));

        QFile cliCertFile(serverCert_);
        sslSocket->setLocalCertificate(QSslCertificate(cliCertFile.readAll()));

        QFile certFile(caCert_);
        sslSocket->addCaCertificate(QSslCertificate(certFile.readAll()));

        sslSocket->setPeerVerifyMode(QSslSocket::VerifyPeer);
        sslSocket->setProtocol(QSsl::SslV3);

        connect(sslSocket, SIGNAL(error(QAbstractSocket::SocketError)),
                this, SLOT(error_handler(QAbstractSocket::SocketError)));
        connect(sslSocket, SIGNAL(sslErrors(QList<QSslError>)),
                this, SLOT(ssl_error_handler(QList<QSslError>)));
        connect(sslSocket, SIGNAL(encrypted()), this,
                SLOT(ready()));
        connect(sslSocket, SIGNAL(readyRead()), this,
                SLOT(read_data_from_client()));

        sslSocket->startServerEncryption();
        if(!sslSocket->waitForEncrypted())
        {
            qDebug() << "failed to perform SSL handshake with client";
            return;
        }
    }

}

void SSLServer::read_data_from_client()
{
    QByteArray qstrbytes = sslSocket->readAll();
    qDebug() << Q_FUNC_INFO << qstrbytes;
}

void SSLServer::ready()
{
    QSslCertificate clientCert = sslSocket->peerCertificate();
    qDebug() << clientCert.isValid();
}

void SSLServer::error_handler(QAbstractSocket::SocketError in_error)
{
    qDebug() << Q_FUNC_INFO << in_error;
    if(in_error == QAbstractSocket::RemoteHostClosedError)
    {
        delete sslSocket; //// line causes crash !!!!!!
    }
}
4

4 回答 4

6

使用QObject::deleteLater()而不是delete因为QSslSocket继承QObject。当您只是delete对象时,您可能仍然会在导致崩溃的套接字上收到消息。

sslSocket->deleteLater();

当您调用 时deleteLater(),Qt 会自动断开所有插槽和信号,并在没有待处理的事件传递给对象后调用对象析构函数。有关QObject::~QObject()更多信息,请参阅。

于 2013-08-27T17:44:39.610 回答
2

如果您认为 QObject 类(如 SSLSocket 类)是如何编写的,它可能是这样的:-

class SSLSocket : public QObject
{        
   signals:
        void sslErrors(QList<QSslError>);

     void SomeFunction()
     {
        // something went wrong, emit error
        emit sslErrors(errorList);

        Cleanup(); // If a slot connected to sslErrors deleted this, what happens now?!
     }
}

当触发信号 sslErrors 时,将调用您的插槽函数。如您所见,发出信号后,该类可能还有更多工作要做。如果你立即删除槽中的对象,这将会崩溃,这就是为什么你应该总是使用 deleteLater() 来删除槽函数中的 QObject 实例。

deleteLater 函数会确保 slot 函数已经执行完毕并且调用栈被恢复,所以会在适当的时候被删除。

请注意,上面的代码实际上并不是 SSLSocket 所做的,而只是一个示例。

于 2013-08-28T10:01:12.670 回答
1

QSslSocket 是一个 QObject。永远不要只删除一个 QObject。当然不要在插槽中执行此操作。始终使用 deleteLater()。

于 2013-08-27T17:43:58.280 回答
1

这是使用 QSslSocket 的 Qt 示例代码:

http://qt-project.org/doc/qt-4.8/network-securesocketclient-sslclient-cpp.html

正如其他海报所提到的,使用deleteLater(),错误通知并不是唯一这样做的地方。

于 2013-08-27T17:50:06.153 回答