QXmlStreamReader/QXmlStreamWriter实现Qt下xml文件读写

版权声明:若无来源注明, Techie亮博客文章均为原创。 转载请以链接形式标明本文标题和地址:
本文标题:QXmlStreamReader/QXmlStreamWriter实现Qt下xml文件读写     本文地址: http://techieliang.com/2017/12/714/

1. 介绍

帮助文档:QXmlStreamReaderQXmlStreamWriter

除此以外读取时还需要使用QXmlStreamAttributes

1.1. QXml-Token标记类型

ConstantValueDescription
QXmlStreamReader::NoToken0The reader has not yet read anything.
QXmlStreamReader::Invalid1An error has occurred, reported in error() and errorString().
QXmlStreamReader::StartDocument2The reader reports the XML version number in documentVersion(), and the encoding as specified in the XML document in documentEncoding(). If the document is declared standalone, isStandaloneDocument() returns true; otherwise it returns false.
QXmlStreamReader::EndDocument3The reader reports the end of the document.
QXmlStreamReader::StartElement4The reader reports the start of an element with namespaceUri() and name(). Empty elements are also reported as StartElement, followed directly by EndElement. The convenience function readElementText() can be called to concatenate all content until the corresponding EndElement. Attributes are reported in attributes(), namespace declarations in namespaceDeclarations().
QXmlStreamReader::EndElement5The reader reports the end of an element with namespaceUri() and name().
QXmlStreamReader::Characters6The reader reports characters in text(). If the characters are all white-space, isWhitespace() returns true. If the characters stem from a CDATA section, isCDATA() returns true.
QXmlStreamReader::Comment7The reader reports a comment in text().
QXmlStreamReader::DTD8The reader reports a DTD in text(), notation declarations in notationDeclarations(), and entity declarations in entityDeclarations(). Details of the DTD declaration are reported in in dtdName(), dtdPublicId(), and dtdSystemId().
QXmlStreamReader::EntityReference9The reader reports an entity reference that could not be resolved. The name of the reference is reported in name(), the replacement text in text().
QXmlStreamReader::ProcessingInstruction10The reader reports a processing instruction in processingInstructionTarget() and processingInstructionData().

主要是用StartDocumentEndDocument文档开始结束,StartElementEndElement元素开始结束、Characters特征

1.2. 范例xml文件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <bookmark href="http://qt-project.org/">
  3. <title>Qt Project</title>
  4. </bookmark>

第一行为StartDocument

bookmark、title为StartElement,/bookmark、/title为EndElement

href为attributes的一项,可以有多项,通过QXmlStreamAttributes::value获取后面的地址内容

Qt Project是title这个element的charcters

2. 写xml

  1. #include <QCoreApplication>
  2. #include <QFile>
  3. #include <QXmlStreamWriter>
  4. int main2(int argc, char *argv[]) {
  5. QCoreApplication a(argc, argv);
  6. QFile file("test.xml");
  7. if(file.open(QIODevice::WriteOnly | QIODevice::Text)) {
  8. QXmlStreamWriter stream(&file);
  9. stream.setAutoFormatting(true);
  10. stream.writeStartDocument();
  11. stream.writeStartElement("bookmark");
  12. stream.writeAttribute("href", "http://qt-project.org/");
  13. stream.writeTextElement("title", "Qt Project");
  14. stream.writeEndElement();
  15. stream.writeEndDocument();
  16. file.close();
  17. }
  18. return 0;
  19. }

写并不复杂,按顺序写即可,注意区分Attribute和Element,如果是一个小节用StartElement,如果是单行的类似于<title>Qt Project</title>可以直接用writeTextElement。

QXmlStreamWriter只是格式操作,并不提供文件操作,需要利用QFile建立文件并传递指针,也可以提供QString的指针,这样最终的xml信息会赋值到QString中。

3. 读xml

  1. #include <QCoreApplication>
  2. #include <QFile>
  3. #include <QXmlStreamReader>
  4. #include <QDebug>
  5. int main(int argc, char *argv[]) {
  6. QCoreApplication a(argc, argv);
  7. QFile file("test.xml");
  8. if(file.open(QIODevice::ReadOnly | QIODevice::Text)) {
  9. QXmlStreamReader xml(&file);
  10. while (!xml.atEnd() && !xml.hasError()) {//循环逐行读取
  11. QXmlStreamReader::TokenType token = xml.readNext();
  12. if(token == QXmlStreamReader::StartDocument)//文件开始跳过
  13. continue;
  14. if(token == QXmlStreamReader::StartElement) {//StartElement类型,主要针对bookmark和title
  15. if(xml.name() == "bookmark") {//bookmark读取,其下attributes有但只有一个
  16. qDebug()<<"StartElement-"<<xml.name();
  17. QXmlStreamAttributes attributes = xml.attributes();
  18. if(attributes.hasAttribute("href")) {//针对href的attribute做判断
  19. qDebug()<<"Attributes-"<<attributes.value("href");//返回值是QStringRef
  20. }
  21. //多个attributes在这里增加更多的if
  22. continue;
  23. }
  24. if(xml.name() == "title") {//title
  25. xml.readNext();//没有attributes,理应直接进入while,此处偷懒了
  26. if(xml.isCharacters()) //可以做Characters判断也可以直接用text
  27. qDebug()<<"title-Characters-"<<xml.text();
  28. }
  29. }
  30. }
  31. if (xml.hasError()) {
  32. //范例,判断枚举类型,写出错误字符串
  33. if(xml.error() == QXmlStreamReader::CustomError) {
  34. //可以直接写出错误字符串
  35. qDebug()<<"error:" <<xml.errorString();
  36. }
  37. }
  38. }
  39. return 0;
  40. }
  • 上述代码都没判断EndElement,不建议这样
  • 建议进入每一个StartElement都直接开启一个while循环,直到循环到EndElement,从何保证对一个开头到结尾的完整操作,而不是像上述代码吧title放在和bookmark同级的循环内,并没有标明title包含于bookmark的关系,这样容易导致错误。
  • 注意xml中每一个<>括住的项都作为一个标记,都占用一次readNext,其开头均为name,其后可能有attribute及对应value。
  • 每两个<>XXX<>之间的XXX均作为一个Characters标记,也占用一次readNext,也就是上文中仍有未打印出的Character(“\n??? “换行符和四个空格、纯换行符):因为QXml以”\r”作为换行识别那么在前后两行的<><>之间还会余留一个\n;在title前的缩进也会当做一项Character。因此建议保证完整的包含关系并做好isXXX或者token枚举类型的判断以免被干扰。

3.1. 其他

除上述xml操作以外,Qt还提供了Qt XML模块,用于更高级的xml操作,提供了高速及使用便利的操作函数(不可兼得呀)

高速的SAX方法读取,QXmlSimpleReader及相关类

使用便捷的DOM方式:QDomDocument及相关类

使用这两个方法,由于是用的非core模块,需要在pro中添加qt += xml

转载于:https://www.cnblogs.com/techiel/p/8052093.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值