Qt中使用DOM对XML进行的各种解析(总结) 转

Qt中使用DOM对XML进行的各种解析(总结)
2009-12-07 11:06

前几天因为底层拓扑结构发生了变化,导致拓扑文件发生了变化,我负责的对于拓扑文件的解析操作自然也就要推到重来。利用上个周末的时间,把这个任务搞定了,顺便也利用这个时间好好总结下这次学到的知识。

我们知道,QT中对于XML文件的读写无非就两种方式,DOM和SAX,两种方法的具体介绍如下:

/***************************************我是传说中的分割线**************************************************************/

用于读取和操作 XML 文件的标准是文档对象模型DOM。 
DOM为 XML 文档的已解析版本定义了一组接口。解析器读入整个文档,然
后构建一个驻留内存的树结构,然后您的代码就可以使用 DOM 接口来操作这个树结构。您可以遍历树以了解原始文档包含了什么,您可以删除树的几个部分,还可以重新排列树和添加新的分支,等等。
遗憾的是,因为DOM 方法涉及读取整个文件并将该文件存储在一个树结构中,而这样可能是低效的、缓
慢的,并且很消耗资源:
DOM 构建整个文档驻留内存的树。如果文档很大,就会要求有极大的内存。 
DOM 创建表示原始文档中每个东西的对象,包括元素、文本、属性和空格。如果您只需关注原始文档的
一小部分,那么创建那些永远不被使用的对象是极其浪费的。
DOM 解析器必须在您的代码取得控制权之前读取整个文档。对于非常大的文档,这会引起显著的延迟。 
这些仅仅是由文档对象模型的设计引起的问题;撇开这些问题,DOM API 是解析 XML 文档非常有用的
方法。

一种替代技术就是SAX。相比于文档对象模型DOM,SAX 是读取和操作 XML 数据的更快速、更轻量的方法。SAX 允许您在读取文档时处理它,从而不必等待整个文档被存储之后才采取操作。它不涉及 DOM 所必需的开销和概念跳跃。

SAX API是一个基于事件的 API,适用于处理数据流,即随着数据的流动而依次处理数据。
SAX API 在其解析您的文档时发生一定事件的时候会通知您。在您对其响应时,您不作保存的数据将会
被抛弃。

/***************************************我是华丽的分割线**********************************************************************/

我们可以看到,如果所要读取的XML文件不是很大,采用DOM读取方法还是很便捷的,由于我用的也是DOM树读取的方法,所以,本文所介绍的也主要是基于DOM的方法读取。

根据常用的操作,我简单的把对XML的操作分为以下几类:

1 首先对XML文件进行操作,打开文件。

这个其实就是对文件进行操作,可以把它直接定义为构造函数,在对对象进行初始化时完成。

TopoDataReader::TopoDataReader(const wstring &filePath):_filePath(filePath),_qDomDoc("mydocument"),_qFile(QString::fromStdWString(filePath))
{
if (!_qFile.open(QIODevice::ReadOnly))
{
   return;
}

if (!_qDomDoc.setContent(&_qFile))
{
   _qFile.close();
   return;
}
}

 

2 读取XML文件中的节点内容

假设XML文件格式如下

(1)

<switchs>

<switch snmpip=211.87.235.136 newwork=front>

</switch>

</switchs>

(2)

<ip>211.87.235.136</ip>

对于第一种情况,采用如下方法:

QDomElement docElem = _qDomDoc.documentElement();

QDomNode nodeswitch=docElem.elementsByTagName("switch ");//红色的为标签名

QDomElement elemnodeswitch=nodeswitch.toElement();

string snmpip=qPrintable(elemnodeswitch.attribute("snmpip"));//network的也是如此方法获取

对于第二种情况,采用如下方法:

直接调用text() API就可以了

string ip=qPrintable(elementnodeip.text());

但是,假设文件中有多个同样的节点,如下

<a>

<b></b>

<b></b>

</a>

这样用elementsByTagName("b")返回的就是一个childNodes()而不是一个单独的node了。

我们可以使用at()方法具体定位。

另外,我们还可以使用这样一种方法获取节点的值,假设XML文件如下

- + < switch snmpIp =" 192.168.120.251 "  network =" front ">
< name > 前端主交换机 </ name >
< description  />
-  < ipList >
< ip > 192.168.120.251 </ ip >
</ ipList >
< rwCommunity > public@120 </ rwCommunity >
< workMode > true </ workMode >
< workStatus > true </ workStatus >
< enableAlarm > true </ enableAlarm >
< snmpCount > 0 </ snmpCount >
< memoryUtilizationRatio > 50.0 </ memoryUtilizationRatio >
< cpuUtilizationRatio > 50.0 </ cpuUtilizationRatio >
< port > 161 </ port >
< snmpStatus > true </ snmpStatus >
< privateName > CZ-5_FA </ privateName >
< switchIndex > topLeft </ switchIndex >
</ switch >

我们可以先获取switch节点,然后得到其childNodes(),于是就可以使用at()方法来获取下面的每个节点了(注:从0开始)

比如说,上面的那个ip节点就是nodeswitch.childNodes().at(3)。

这样做得好处就是不用担心重复的问题,只要你确定switch节点确定对了,底下的也就一定能确定了。

3 返回某个节点下子节点的个数

这个简单,也是直接调API

QDomElement docElem = _qDomDoc.documentElement();
QDomNode nodetagname=docElem.elementsByTagName(tagname).at(0);//假设有多个tagname此处选第一个
int num=nodetagname.childNodes().size();

好了,临时想起来就这么多。以后想起来再添。

 

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值