在我的应用程序中,我从本地(非Unicode)字符集中的文件中读取字符串字段。该字段是10个字节,如果字符串<10个字节,则余数用零填充。
char str ="STRING\0\0\0\0"; // that was read from file
QByteArray fieldArr(str,10); // fieldArr now is STRING\000\000\000\000
fieldArr = fieldArr.trimmed() // from some reason array still containts zeros
QTextCodec *textCodec = QTextCodec::codecForLocale();
QString field = textCodec->ToUnicode(fieldArr).trimmed(); // also not removes zeros
所以我的问题 - 如何从字符串中删除尾随零?
附:调试时,我在“Local and Expressions”窗口中看到了零
我将假设str
应该是char const *
而不是char
。
只是不要超过QByteArray
- QTextCodec
可以处理一个C字符串,它以第一个空字节结束:
QString field = textCodec->toUnicode(str).trimmed();
附录:由于字符串可能不是零终止,因此将空字节的存储添加到最后似乎是不可能的,并且制作副本以准备复制似乎是浪费,我建议自己计算长度并使用toUnicode
重载接受char指针和长度。
std::find
对此有好处,因为如果在其中找不到元素,它将返回给定范围的结束迭代器。这使得不需要特殊情况处理:
QString field = textCodec->toUnicode(str, std::find(str, str + 10, '\0') - str).trimmed();
我会这样做:
char* str = "STRING\0\0\0\0";
QByteArray fieldArr;
for(quint32 i = 0; i < 10; i++)
{
if(str[i] != '\0')
{
fieldArr.append(str[i]);
}
}
QString
可以使用fromLocal8Bit从char数组指针构造。选择编解码器的方式与在代码中手动操作的方式相同。
您需要手动将长度设置为10,因为您说您无法保证存在终止空字节。
然后你可以使用remove()
去除所有空字节。警告:STRI\0\0\0\0NG
也将导致STRING
,但你说这不会发生。
char *str = "STRING\0\0\0\0"; // that was read from file
QString field = QString::fromLocal8Bit(str, 10);
field.remove(QChar::Null);
这对你有用吗?
#include <QDebug>
#include <QByteArray>
int main()
{
char str[] = "STRING\0\0\0\0";
auto ba = QByteArray::fromRawData(str, 10);
qDebug() << ba.trimmed(); // does not work
qDebug() << ba.simplified(); // does not work
auto index = ba.indexOf('\0');
if (index != -1)
ba.truncate(index);
qDebug() << ba;
return 0;
}
使用fromRawData()
可以节省额外的副本。确保str
保持不变,直到你删除ba
。
indexOf()
是安全的,即使你已经填满整个str
,因为QByteArray
知道你只有10个字节,你可以安全访问。它不会触及11或更晚。没有缓冲区溢出。
一旦你删除了额外的\0
,转换为QString
是微不足道的。
您可以在第一个\0
之后截断字符串:
char * str = "STRING\0\0\0\0"; // Assuming that was read from file
QString field(str); // field == "STRING\0\0\0\0"
field.truncate(field.indexOf(QChar::Null)); // field == "STRING" (without '\0' at the end)