3

我正在使用 Qt 开发 NMDC 客户端(p2p、DC++ 和朋友)。协议本身非常简单:

$command parameters|

除了压缩

“ZPipe 通过向客户端发送命令 $ZOn| 来工作。在 $ZOn 之后,将跟随一个包含命令的 ZLib 压缩流。该流将以 ZLib 定义的 EOF 结束。(压缩流中没有 $ZOff!)”

以下是相关代码:

QTcpSocket *conn;
bool compressed;
QByteArray zbuffer;
QByteArray buffer;

// ...

void NMDCConnection::on_conn_readyRead() {
    // this gets called whenever we get new data from the hub

    if(compressed) {            // gets set when we receive $ZOn
        zbuffer.append(conn->readAll());


        // Magic happens here


        if( stream_is_complete ) {
            buffer.append(uncompressed_stream);
            buffer.append(remainder_of_data);
            compressed = false;
        }
    } else { 
        buffer.append(conn->readAll());
    };
    parse(buffer);
}

那么,如何获得stream_is_completeuncompressed_stream和的值remainder_of_data?我无法寻找下一个'$',因为流可以包含它。我尝试在 zlib 文档中寻找类似于 EOF 的东西,但没有这样的东西,事实上,每个流都以看似随机的字符结尾。

我也玩过 qUncompress(),但它需要一个完整的流,仅此而已。

4

2 回答 2

1

你是直接用zlib吗?

完全未经测试...

z_stream zstrm;
QByteArray zout;
// when you see a $ZOn|, initialize the z_stream struct
parse() {
    ...
    if (I see a $ZOn|) {
        zstrm.next_in = Z_NULL;
        zstrm.avail_in = 0;
        zstrm.zalloc = Z_NULL;
        zstrm.zfree = Z_NULL;
        zstrm.opaque = 0;
        inflateInit(&zstrm);
        compressed = true;
    }
}
void NMDCConnection::on_conn_readyRead() {
    if (compressed) {
        zbuffer.append(conn->readAll());
        int rc;
        do {
            zstrm.next_in = zbuffer.data();
            zstrm.avail_in = zbuffer.size();
            zout.resize(zstrm.total_out + BLOCK_SIZE);
            zstrm.next_out = zout.data() + zstrm.total_out;
            zstrm.avail_out = BLOCK_SIZE;
            rc = inflate(&zstrm, Z_SYNC_FLUSH);
            zbuffer.remove(0, zstrm.next_in - zbuffer.data());
        } while (rc == Z_OK && zstrm->avail_out == 0);
        if (rc == Z_STREAM_END) {
            zout.truncate(zstrm.total_out);
            buffer.append(zout);
            zout.clear();
            buffer.append(zbuffer);
            zbuffer.clear();
            compress = false;
            inflateEnd(&zstrm);
        }
        else if (rc != Z_OK) {
            // ERROR!  look at zstrm.msg
        }
    }
    else // whatever
}

qbuffer这逐渐从to解压缩(膨胀) ,并在说“不再”qout时停止。inflate

也许不如从QuaZip 借来

于 2009-10-28T16:39:32.260 回答
0

文件结尾对于 zlib 来说并不特殊。我在您的代码中看到的问题是您使用“readAll()”,它实际上没有任何报告“错误”的方法,例如文件结尾。

您应该尝试在循环中使用“read()”。如果您读取流并返回读取的 0 字节,则可以确定您已到达流的末尾(另一端已关闭连接)。您已读取的缓冲区,加入循环中所有先前“读取”的缓冲区,将为您提供完整的缓冲区。

于 2009-10-28T15:29:48.660 回答