Qt 读写 XML

bool MAS_ParseBase::setContent( QDomDocument& document )
{
	if ( !document.setContent( m_pFile, false ) )
	{
		return false;
	}
	m_pFile->close();
}


void MAS_ParseBase::save( QDomDocument& document )
{
	QTextStream out( m_pFile );
	out.setCodec( "UTF-8" );
	document.save( out, 4 );
	m_pFile->close();
}


void MAS_ParseBase::getSubNode( QDomElement element, XMLNode& xml )
{
	// 循环记录子节点信息
	QDomNode firstNode = element.firstChild();
	while ( !firstNode.isNull() )
	{
		QDomElement firstElement = firstNode.toElement();

		XMLNode xmlFirst;
		xmlFirst.qstrTagName = firstElement.tagName();
		QDomNamedNodeMap nodeMap = firstElement.attributes();
		for ( int i=0; i<nodeMap.size(); i++ )
		{
			NodeAttribute attribute;
			attribute.qstrNodeName = nodeMap.item(i).nodeName();
			attribute.qstrNodeValue = nodeMap.item(i).nodeValue();
			xmlFirst.vtAttribute.push_back(attribute);
		}

		if ( !firstElement.firstChild().toElement().isNull() )
		{
			getSubNode( firstElement, xmlFirst );
		}
		else
		{
			xmlFirst.qstrValue = firstElement.toElement().text();
		}
		
		xml.vtSubNode.push_back( xmlFirst );

		firstNode = firstNode.nextSibling();
	}
}


void MAS_ParseBase::setSubNode( QDomDocument& document, QDomElement element, std::vector< XMLNode >& vtXML )
{
	for ( int i=0; i<vtXML.size(); i++ )
	{
		// 添加子节点属性信息
		QDomElement firstElement = document.createElement( vtXML.at(i).qstrTagName );
		for ( int m=0; m<vtXML.at(i).vtAttribute.size(); m++ )
		{
			firstElement.setAttribute( vtXML.at(i).vtAttribute.at(m).qstrNodeName, vtXML.at(i).vtAttribute.at(m).qstrNodeValue );
		}
		element.appendChild( firstElement );

		if ( vtXML.at(i).vtSubNode.size() > 0 )
		{
			setSubNode( document, firstElement, vtXML.at(i).vtSubNode );
		}
		else
		{
			firstElement.appendChild( document.createTextNode( vtXML.at(i).qstrValue ) );
		}
	}
}


XMLNode MAS_ParseXML::read()
{
	XMLNode xml;
	QDomDocument document;
	if ( !setContent(document) )
	{
		return xml;
	}

	QDomElement element = document.documentElement();

	// 记录根节点信息
	xml.qstrTagName = element.tagName();

	QDomNamedNodeMap nodeMap = element.attributes();
	for ( int i=0; i<nodeMap.size(); i++ )
	{
		NodeAttribute attribute;
		attribute.qstrNodeName = nodeMap.item(i).nodeName();
		attribute.qstrNodeValue = nodeMap.item(i).nodeValue();
		xml.vtAttribute.push_back(attribute);
	}

	// 递归获取子节点
	getSubNode( element, xml );

	return xml;
}

void MAS_ParseXML::write( XMLNode xml )
{
	// 添加文件头信息
	QDomDocument document;
	QDomProcessingInstruction instruction = document.createProcessingInstruction( "xml", "version = \"1.0\" encoding = \"UTF-8\"" );
	document.appendChild( instruction );

	// 添加根节点
	QDomElement element = document.createElement( xml.qstrTagName );
	for ( int i=0; i<xml.vtAttribute.size(); i++ )
	{
		element.setAttribute( xml.vtAttribute.at(i).qstrNodeName, xml.vtAttribute.at(i).qstrNodeValue );
	}
	document.appendChild( element );

	// 递归写入子节点
	setSubNode( document, element, xml.vtSubNode );

	// 将数据写入到XML文件,文件的每个空间为4个空格
	save( document );

	QMessageBox::information( NULL, QString::fromLocal8Bit("文件写入进度提示:"), QString::fromLocal8Bit("文件写入完毕!"), QMessageBox::Ok );
}


XML解析用到的结构体:

struct NodeAttribute
{
	QString qstrNodeName;
	QString qstrNodeValue;
};
struct XMLNode
{
	QString qstrTagName;
	QString qstrValue;
	std::vector< NodeAttribute > vtAttribute;
	std::vector< XMLNode > vtSubNode;
};
<think>好的,我现在需要回答用户关于如何使用Qt进行XML文件读写的问题。用户提到他们想看到示例教程,所以我应该提供具体的步骤和代码示例。首先,我需要回忆一下Qt中处理XML的两种主要方法:DOM和基于流的解析(比如QXmlStreamReader和QXmlStreamWriter)。 根据用户提供的引用内容,引用[1]和引用[2]分别提到了QXmlStreamReader和QDomDocument的使用。用户的问题可能希望得到这两种方法的示例。我需要先解释这两种方法的区别,然后分别给出读写操作的示例代码。 首先,DOM方式会将整个XML文档加载到内存中,适合处理较小的文件,操作起来比较方便。而基于流的解析则是逐行读取,内存消耗小,适合大文件。我应该分别给出这两种方法的读写示例。 对于DOM方式,写文件时需要使用QDomDocument、QDomElement等类来创建节点,然后写入文件。读文件时同样使用QDomDocument解析整个文档,遍历节点。例如,创建根元素、子元素,设置属性等步骤。 对于流方式,写文件使用QXmlStreamWriter,按顺序写入元素;读文件使用QXmlStreamReader,逐个读取元素并处理。需要检查元素类型,如StartElement、EndElement,并获取元素名称和内容。 用户提供的引用中,引用[1]的示例使用了QXmlStreamReader,而引用[2]提到了QDomDocument和QXmlSimpleReader。但用户的问题更关注读写操作,所以重点放在DOM和流式两种方式。 我需要确保代码示例正确,比如在写文件时正确打开文件并设置编码,读文件时处理可能出现的错误。同时,注意关闭文件,释放资源。 另外,需要指出两种方式的优缺点,比如DOM适合小文件,流式适合大文件,帮助用户根据需求选择合适的方法。 最后,生成相关问题,可能涉及性能比较、错误处理、其他格式的区别等。例如,如何选择DOM和流式解析,或者处理XML中的特殊字符等。 现在,我需要将以上思路整理成结构化的回答,分步骤讲解,并提供代码示例。确保使用中文,符合用户的要求,并在适当位置添加引用标记。</think>### Qt XML文件读写操作教程 Qt提供DOM和流式解析两种XML处理方式,以下为具体实现方法: #### 一、DOM方式(适合小文件) **写入XML文件示例** ```cpp // 创建XML文档对象 QDomDocument doc; QDomProcessingInstruction header = doc.createProcessingInstruction("xml", "version='1.0' encoding='UTF-8'"); doc.appendChild(header); // 创建根元素 QDomElement root = doc.createElement("Settings"); doc.appendChild(root); // 添加子节点 QDomElement resolution = doc.createElement("Resolution"); resolution.setAttribute("width", 1920); resolution.setAttribute("height", 1080); root.appendChild(resolution); // 写入文件 QFile file("config.xml"); file.open(QIODevice::WriteOnly); QTextStream stream(&file); doc.save(stream, 4); file.close(); ``` **读取XML文件示例** ```cpp QFile file("config.xml"); if (!file.open(QIODevice::ReadOnly)) return; QDomDocument doc; if (!doc.setContent(&file)) { file.close(); return; } QDomElement root = doc.documentElement(); QDomNode node = root.firstChild(); while (!node.isNull()) { if (node.isElement()) { QDomElement elem = node.toElement(); if (elem.tagName() == "Resolution") { int width = elem.attribute("width").toInt(); int height = elem.attribute("height").toInt(); qDebug() << "Resolution:" << width << "x" << height; } } node = node.nextSibling(); } file.close(); ``` #### 二、流式解析(适合大文件) **写入XML示例** ```cpp QFile file("data.xml"); file.open(QIODevice::WriteOnly); QXmlStreamWriter writer(&file); writer.setAutoFormatting(true); writer.writeStartDocument(); writer.writeStartElement("Students"); writer.writeStartElement("Student"); writer.writeAttribute("id", "1001"); writer.writeTextElement("Name", "张三"); writer.writeTextElement("Score", "95"); writer.writeEndElement(); // Student writer.writeEndElement(); // Students writer.writeEndDocument(); file.close(); ``` **读取XML示例** ```cpp QFile file("data.xml"); file.open(QIODevice::ReadOnly); QXmlStreamReader reader(&file); while (!reader.atEnd()) { if (reader.isStartElement()) { if (reader.name() == "Student") { QString id = reader.attributes().value("id").toString(); qDebug() << "学号:" << id; } else if (reader.name() == "Name") { qDebug() << "姓名:" << reader.readElementText(); } else if (reader.name() == "Score") { qDebug() << "分数:" << reader.readElementText(); } } reader.readNext(); } if (reader.hasError()) { qDebug() << "解析错误:" << reader.errorString(); } file.close(); ``` #### 三、方法对比 | 特性 | DOM方式 | 流式解析 | |---------------------|---------------------------|-------------------------| | 内存占用 | 高(加载整个文档) | 低(逐行处理) | | 适用场景 | 小型文件/需要随机访问 | 大型文件/顺序处理 | | 实现复杂度 | 简单直观 | 需要状态管理 | | 修改能力 | 支持修改和保存 | 仅支持读取或顺序写入 | [^1]: DOM方式通过树形结构操作XML节点,适合配置文件的读写 [^2]: 流式解析采用事件驱动模型,处理大文件时性能更优
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值