【问题标题】:Passing QByteArray containing binary data to function by value传递包含二进制数据的 QByteArray 以按值运行
【发布时间】:2023-03-14 19:10:01
【问题描述】:

所以我正在开发一个大型程序来操作存储在 QByteArrays 中的二进制数据。今天,我意识到当我将 QByteArray 按值传递给函数时,例如

void myFunc(QByteArray data)
{
 // Do stuff
}

如果我的 QByteArray 在二进制流中间的某处有一个 0x0,比如0xA5 0x20 0x04 0x00 0x52。将 QByteArray 传递给函数时,0x52 被截断。

我想我可以简单地将 QByteArray 按值传递给函数,但在某些情况下我需要数组的副本(用于数据持久性或添加到队列等),不幸的是,我有很多,许多代码行,这些函数在复制的 QByteArrays 周围传递。

所以我想知道是否有任何方法可以重载赋值运算符(或使用其他方法)以在两个 QByteArrays 之间复制数据而不将 0x00 项视为以空字符结尾的字符串字符。我试图找到解决这个问题的最简单方法,而不必完全破坏我的程序并重做所有功能。有没有更好的替代数据类型可以使用(我可以在其中快速查找/替换,或者我可以将它们包装在某些函数中?

更新:为演示问题添加了精简示例

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    QByteArray data = QByteArrayLiteral("\xA5\x20\x04\x00\x52");

    doStuff(data);

    return a.exec();
}

void doStuff(QByteArray theData)
{
    // theData will contain only 0xA5 0x20 0x04 0x00 if you look from in here
}

更新 2:

好吧,显然,在测试了上面的例子之后。它实际上确实有效。但是,我在线程中使用的真实场景需要进行更多测试。一旦我进一步缩小问题范围,将更新...

【问题讨论】:

  • 如果您的 MRE 没有重现问题,那么它不是 MRE,因此您必须努力解决。

标签: c++ qt qbytearray


【解决方案1】:

我认为这与通过值传递 QByteArray 没有任何关系,但更像是您如何初始化数组。考虑这个例子:

QByteArray data("\xA5\x20\x04\x00\x52");

qDebug( data.toHex() );

QByteArray data2;
data2.append( static_cast<char>( 0xA5 ) );
data2.append( static_cast<char>( 0x20 ) );
data2.append( static_cast<char>( 0x04 ) );
data2.append( static_cast<char>( 0x00 ) );
data2.append( static_cast<char>( 0x52 ) );

qDebug( data2.toHex() );

输出是:

a52004
a520040052

用它在 0 终止处截断的字符串初始化它。这也适用于QByteArray::toStdString()

编辑:

正如弗兰克指出的那样:

QByteArray data("\xA5\x20\x04\x00\x52", 5);

qDebug( data.toHex() );

确实会推出预期的内容:

a520040052

【讨论】:

  • 这是因为构造函数有一个参数size,默认为-1。然后“如果大小为负,则假定数据指向以'\0'结尾的字符串,并且其长度是动态确定的。终止\0字符不被视为字节数组的一部分”。因此,如果您希望将 nul 字节视为普通字符,则需要显式传递数组大小。
  • 谢谢。我也遇到了正确打印 QByteArray 中的字节的问题,所以这非常有用!
猜你喜欢
  • 2012-06-10
  • 2013-05-27
  • 2016-09-01
  • 1970-01-01
  • 1970-01-01
  • 2014-10-05
  • 1970-01-01
  • 1970-01-01
  • 2011-07-26
相关资源
最近更新 更多