1

早上好,我正在寻找一个关于使用 QTcpSocket 将文件从一台电脑发送到另一台电脑的示例。我试图创建自己的代码。我有一个应用程序,其中,用户将从他的 DD 中选择一个文件(所有类型)并将其发送到 TcpServer,然后该服务器将此文件发送到其他客户端。但是,当我选择时,我有一个问题文件和我发送它,在客户端,我有这个消息:文件正在发送,但在服务器端,它告诉我文件没有收到它的全部字节。请有任何建议。这是在客户端发送文件的函数:

void FenClient::on_boutonEnvoyer_2_clicked()
{
    QString nomFichier = lineEdit->text();
        QFile file(lineEdit->text());
        if(!file.open(QIODevice::ReadOnly))
        {
            qDebug() << "Error, file can't be opened successfully !";
            return;

        }

        QByteArray bytes = file.readAll();

        QByteArray block;
        QDataStream out(&block, QIODevice::WriteOnly);

        out << quint32(0);
        out << nomFichier;
        out << bytes;
        out.device()->seek(0);
        out << quint32((block.size() - sizeof(quint32)));

        qDebug() << "Etat : envoi en cours...";
         listeMessages->append("status : sending the file...");

        socket->write(block);
}

和服务器端:

void FenServeur::datarecieved()
{

    QTcpSocket *socket = qobject_cast<QTcpSocket *>(sender());

        if(socket == 0)
        {
            qDebug() << "no Socket!";
            return;
        }

        forever
        {
            QDataStream in(socket);
            if(blockSize == 0)
            {
                if(socket->bytesAvailable()  )
                {
                    qDebug() << "Error < sizeof(quint32))";
                    return;
                }

                in >> blockSize;
            }

            if(socket->bytesAvailable() < blockSize)
             {
                qDebug() << "data not recieved with its total bytes";

                return;
             }

            qDebug() << "!!!!!!";
            QByteArray dataOut;
            QString nameFile;
            in >> nameFile >> dataOut;
            QFile fileOut(nameFile);
            fileOut.open(QIODevice::WriteOnly);
            fileOut.write(dataOut);
            fileOut.close();

            blockSize = 0;
         }
}

void FenServeur::sendToAll(const QString &message)
{

    QByteArray paquet;
    QDataStream out(&paquet, QIODevice::WriteOnly);

    out << (quint32) 0;
    out << message;
    out.device()->seek(0);
    out << (quint32) (paquet.size() - sizeof(quint32));
    for (int i = 0; i < clients.size(); i++)
    {
        clients[i]->write(paquet);
    }

}

所以我无法将服务器收到的文件写入新文件。请有任何建议!!并提前感谢

4

1 回答 1

0

你的代码在等对方,而对方却在等你。任何允许双方相互等待的协议都从根本上被破坏了。

TCP 允许发送方等待接收方,但不允许接收方等待发送方。这是有道理的,因为不允许发送者等待接收者需要无限量的缓冲。因此,对于任何在 TCP 之上的应用程序,接收者可能不会等待发送者。

但你做了:

        if(socket->bytesAvailable() < blockSize)
        {
            qDebug() << "data not recieved with its total bytes";

            return;
        }

在这里,您正在等待发送方取得进展(bytesAvailable 增加),然后您才愿意接收(从套接字拉数据)。但是发送者在它愿意发送更多数据之前正在等待你取得进展。这会导致死锁。不要这样做。

尽可能快地接收尽可能多的数据。在从网络堆栈中提取已经接收到的数据之前,切勿坚持通过网络接收更多数据。

于 2012-09-29T08:38:43.030 回答