【问题标题】:QBtyeArray QDataStream qCompress to file adding extra leading bytesQBtyeArray QDataStream qCompress 到文件添加额外的前导字节
【发布时间】:2021-06-03 16:07:34
【问题描述】:

Qt/C++ 程序具有将对象数据 (_token) 写入文件的函数,如下所示:

    QFile f( _tokenFile);

    if (!f.open( QIODevice::WriteOnly)) {
        qDebug() << "Unable to open token file for writing" << f.errorString();
        return false;
    }

    QByteArray tokenBa;
    QDataStream ds( &tokenBa, QIODevice::WriteOnly);
    ds << _token;

    tokenBa = qCompress( tokenBa);

    f.write( tokenBa);
    f.close();

_token 是以下struct 的一个实例:

struct Token {
    QString accessToken;
    QString refreshToken;
    QString clientSecret;
    QString authCode;
    QTime expiryTime;


    enum AuthState {
        A_Invalid,
        A_RequestAuth,
        A_Authenticated,
    };

    AuthState state;

    Token() : state( A_Invalid) {}

    bool isValid() const {
        if (accessToken.isEmpty() ||
            refreshToken.isEmpty()) {
            return false;
        }
        return true;
    }

    void inValidate() {
        accessToken.clear();
        refreshToken.clear();
        clientSecret.clear();
        authCode.clear();
        expiryTime = QTime();
    }

    void cleanUp() {
        accessToken.clear();
        refreshToken.clear();
        clientSecret.clear();
        authCode.clear();
        expiryTime = QTime();
    }
};

当文件被保存时,它在开始时有 4 个额外的字节,这会将文件呈现为无效的 zlib 文件。


0000000 0000 5e01 9c78 904d 4f5d 5082 871c fa9f
0000020 353e 25cd 6975 2c2f d563 4c2c 62b8 cad1

我们可以看到上面的字节 5-6 是 9C 78 这是 zlib 签名,但在这之前的 4 个字节是问题。

要检查压缩数据是否正确,我执行以下操作:

dd if=file.token bs=1 skip=4 | openssl zlib -d

这会产生预期的结果(用于测试)。

问题在于应用程序将这些数据读回数据对象:

    QFile f( _tokenFile);

    if (!f.exists()) {
        qDebug() << "Token file doesn't exist"  << f.fileName();
        return false;
    }

    if (!f.open( QIODevice::ReadOnly)) {
        qDebug() << "Unable to open token file for reading" << f.errorString();
        return false;
    }

    QByteArray tokenBa = f.readAll();
    f.close();

    if (tokenBa.isEmpty()) {
        qDebug() << "Token file is empty.";
        return false;
    }

    tokenBa = qUncompress( tokenBa);

    QDataStream ds( &tokenBa, QIODevice::ReadOnly);
    ds >> _token;

这将返回 null - 因为前导 4 个无关字节。我可以输入一些代码来跳过这 4 个前导字节,但我怎么知道它总是 4 个字节?我想肯定的是文件数据都是 zlib 压缩的。

我的问题是如何避免首先保存这些字节,以便在重新读取时知道格式是 zlib 类型?

【问题讨论】:

    标签: qt zlib qbytearray qdatastream


    【解决方案1】:

    您无法避免它们,因为稍后qUncompress 需要它们:

    注意:如果要使用此函数解压缩使用 zlib 压缩的外部数据,首先需要在包含数据的字节数组中添加一个四字节的标头。标头必须包含未压缩数据的预期长度(以字节为单位),以无符号、大端序、32 位整数表示。

    【讨论】:

    • 谢谢。所以在我的例子中,前 4 个字节 0000 5e01 应该正是这样做的,这就是解压缩失败的原因。
    • 0x5e01 是 24065 - 因此您的文件应该是 24069 字节大并且您读入的 QByteArray - 是这样吗?
    • 没有。实际大小约为 240 字节。也许问题在于文件的保存。我会做更多的调试和检查。
    • 然后添加debug语句,在写文件的时候打印出tokenBA的大小。并且可能还在那里输出 bytearray 的前 4 个字节,以查看它们是否包含预期的字节。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-06-16
    • 1970-01-01
    • 2010-11-13
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多