您的位置:首页 > 编程语言 > Qt开发

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

2010-09-07 10:59 591 查看
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();

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

 

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  xml qt 文档 api string network