Qt 学习之路 2(37):文本文件读写

本文介绍了Qt中如何使用QTextStream进行文本文件的读写操作,包括自动的编码转换和换行符处理,以及QIODevice和QFile的使用。通过示例展示了如何以文本格式写入和读取数据,并讨论了不同打开方式的影响。还提到了QTextStream的编码设置和stream manipulators用于格式化的输出。
摘要由CSDN通过智能技术生成

上一章我们介绍了有关二进制文件的读写。二进制文件比较小巧,却不是人可读的格式。而文本文件是一种人可读的文件。为了操作这种文件,我们需要使用QTextStream类。QTextStreamQDataStream的使用类似,只不过它是操作纯文本文件的。另外,像 XML、HTML 这种,虽然也是文本文件,可以由QTextStream生成,但 Qt 提供了更方便的 XML 操作类,这里就不包括这部分内容了。

QTextStream会自动将 Unicode 编码同操作系统的编码进行转换,这一操作对开发人员是透明的。它也会将换行符进行转换,同样不需要自己处理。QTextStream使用 16 位的QChar作为基础的数据存储单位,同样,它也支持 C++ 标准类型,如 int 等。实际上,这是将这种标准类型与字符串进行了相互转换。

QTextStreamQDataStream的使用基本一致,例如下面的代码将把“The answer is 42”写入到 file.txt 文件中:

这里,我们在open()函数中增加了QIODevice::Truncate打开方式。我们可以从下表中看到这些打开方式的区别:

枚举值描述
QIODevice::NotOpen未打开
QIODevice::ReadOnly以只读方式打开
QIODevice::WriteOnly以只写方式打开
QIODevice::ReadWrite以读写方式打开
QIODevice::Append以追加的方式打开,新增加的内容将被追加到文件末尾
QIODevice::Truncate以重写的方式打开,在写入新的数据时会将原有数据全部清除,游标设置在文件开头。
QIODevice::Text在读取时,将行结束符转换成 \n;在写入时,将行结束符转换成本地格式,例如 Win32 平台上是 \r\n
QIODevice::Unbuffered忽略缓存

我们在这里使用了QFile::WriteOnly | QIODevice::Truncate,也就是以只写并且覆盖已有内容的形式操作文件。注意,QIODevice::Truncate会直接将文件内容清空。

虽然QTextStream的写入内容与QDataStream一致,但是读取时却会有些困难:

在使用QDataStream的时候,这样的代码很方便,但是使用了QTextStream时却有所不同:读出的时候,str 里面将是 The answer is 42,ans 是 0。这是因为以文本形式写入数据,是没有数据之间的分隔的。还记得我们前面曾经说过,使用QDataStream写入的时候,实际上会在要写入的内容前面,额外添加一个这段内容的长度值。而文本文件则没有类似的操作。因此,使用文本文件时,很少会将其分割开来读取,而是使用诸如QTextStream::readLine()读取一行,使用QTextStream::readAll()读取所有文本这种函数,之后再对获得的QString对象进行处理。

默认情况下,QTextStream的编码格式是 Unicode,如果我们需要使用另外的编码,可以使用

这样的函数进行设置。

另外,为方便起见,QTextStreamstd::cout一样提供了很多描述符,被称为 stream manipulators。因为文本文件是供人去读的,自然需要良好的格式(相比而言,二进制文件就没有这些问题,只要数据准确就可以了)。这些描述符是一些函数的简写,我们可以从文档中找到:

描述符等价于
binsetIntegerBase(2)
octsetIntegerBase(8)
decsetIntegerBase(10)
hexsetIntegerBase(16)
showbasesetNumberFlags(numberFlags() | ShowBase)
forcesignsetNumberFlags(numberFlags() | ForceSign)
forcepointsetNumberFlags(numberFlags() | ForcePoint)
noshowbasesetNumberFlags(numberFlags() & ~ShowBase)
noforcesignsetNumberFlags(numberFlags() & ~ForceSign)
noforcepointsetNumberFlags(numberFlags() & ~ForcePoint)
uppercasebasesetNumberFlags(numberFlags() | UppercaseBase)
uppercasedigitssetNumberFlags(numberFlags() | UppercaseDigits)
lowercasebasesetNumberFlags(numberFlags() & ~UppercaseBase)
lowercasedigitssetNumberFlags(numberFlags() & ~UppercaseDigits)
fixedsetRealNumberNotation(FixedNotation)
scientificsetRealNumberNotation(ScientificNotation)
leftsetFieldAlignment(AlignLeft)
rightsetFieldAlignment(AlignRight)
centersetFieldAlignment(AlignCenter)
endloperator<<('\n')flush()
flushflush()
resetreset()
wsskipWhiteSpace()
bomsetGenerateByteOrderMark(true)

这些描述符只是一些函数的简写。例如,我们想要输出 12345678 的二进制形式,那么可以直接使用

就可以了。这等价于

更复杂的,如果我们想要舒服 1234567890 的带有前缀、全部字母大写的十六进制格式(0xBC614E),那么只要使用

即可。

不仅是QIODeviceQTextStream也可以直接把内容输出到QString。例如

这提供了一种简单的处理字符串内容的方法。


=============================================================
===源博客地址:https://www.devbean.net/2012/08/qt-study-road-2-catelog/===
=============================================================

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值