加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

TinyXml 优秀的开源 xml 解析器

发布时间:2020-12-16 09:43:04 所属栏目:百科 来源:网络整理
导读:TinyXml 优秀的开源 xml 解析器 2010-08-11 11:10:04 分类: C/C++ 读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似,面向对象性很好。 TinyXML是一个 开源 的解析XML
TinyXml 优秀的开源 xml 解析器 2010-08-11 11:10:04

分类:C/C++

读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似,面向对象性很好。

TinyXML是一个开源的解析XML的解析库,能够用于C++,能够在Windows或Linux中编译。这个解析库的模型通过解析XML文件,然后在内存中生成DOM模型,从而让我们很方便的遍历这棵XML树。

DOM模型即文档对象模型,是将整个文档分成多个元素(如书、章、节、段等),并利用树型结构表示这些元素之间的顺序关系以及嵌套包含关系。

如下是一个XML片段:

代码链接:TinyXml.zip

<Persons>
<Person ID="1"<name>周星星</name<age>20/age/Person"2">白晶晶>18/Persons>

在TinyXML中,根据XML的各种元素来定义了一些类:

TiXmlBase:整个TinyXML模型的基类。

TiXmlAttribute:对应于XML中的元素的属性

TiXmlNode:对应于DOM结构中的节点

TiXmlComment:对应于XML中的注释

TiXmlDeclaration:对应于XML中的申明部分,即<?versiong="1.0"?>。

TiXmlDocument:对应于XML的整个文档。

TiXmlElement:对应于XML的元素。

TiXmlText:对应于XML的文字部分

TiXmlUnknown:对应于XML的未知部分。

TiXmlHandler:定义了针对XML的一些操作。

TinyXML是个解析库,主要由DOM模型类(TiXmlBase、TiXmlNode、TiXmlAttribute、TiXmlComment、TiXmlDeclaration、TiXmlElement、TiXmlText、TiXmlUnknown)和操作类(TiXmlHandler)构成。它由两个头文件(.h文件)和四个CPP文件(.cpp文件)构成,用的时候,只要将(tinyxml.h、tinystr.h、tinystr.cpp、tinyxml.cpp、tinyxmlerror.cpp、tinyxmlparser.cpp)导入工程就可以用它的东西了。如果需要,可以将它做成自己的DLL来调用。举个例子就可以说明一切。。。

对应的XML文件:

>phinecos>22>


读写XML文件的程序代码:

#include<iostream>
#include"tinyxml.h"
#include"tinystr.h"
#include<string>
#include<windows.h<atlstr>
using namespace std;

CString GetAppPath()
{//获取应用程序根目录
TCHAR modulePath[MAX_PATH];
GetModuleFileName(NULL,modulePath);
CString strModulePath(modulePath;
strModulePath=strModulePath.Left(strModulePath.ReverseFind(_T('';
return strModulePath;
}


/创建xml文件若创建成功返回true(string&szFileName{
try
{
/创建一个XML的文档对象。
TiXmlDocument*myDocument=newTiXmlDocument/创建一个根元素并连接。
TiXmlElement*RootElement=newTiXmlElement"Persons";
myDocument->LinkEndChild(RootElement;


/创建一个Person元素并连接。
TiXmlElement*PersonElement=new TiXmlElement"Person";
RootElement->LinkEndChild(PersonElement/设置Person元素的属性。
PersonElement->SetAttribute"ID"/创建name元素、age元素并连接。
TiXmlElement*NameElement"name";
TiXmlElement*AgeElement"age";
PersonElement-(NameElement(AgeElement/设置name元素和age元素的内容并连接。
TiXmlText*NameContent=new TiXmlText"周星星";
TiXmlText*AgeContent"22";
NameElement-(NameContent;
AgeElement-(AgeContent;
CString appPath=GetAppPath;
string seperator"";
string fullPath=appPath.GetBuffer(0+seperator+szFileName>SaveFile(fullPath.c_str;/保存到文件
}
catch&e{
return false}
return true/读取Xml文件,并遍历
bool ReadXmlFile{
CString appPath/创建一个XML的文档对象。
TiXmlDocument=new TiXmlDocument>LoadFile/获得根元素,即Persons。
TiXmlElement=myDocument->RootElement/输出根元素名称,即输出Persons。
cout<RootElement->Value<endl/获得第一个Person节点。
TiXmlElement*FirstPerson=RootElement->FirstChildElement/获得第一个Person的name节点和age节点和ID属性。
TiXmlElement=FirstPerson->FirstChildElement;// 结点
TiXmlElement=NameElement->NextSiblingElement;//结点
TiXmlAttribute*IDAttribute>FirstAttribute;// 属性
/输出第一个Person的name内容,即周星星;age内容,即;ID属性,即。
cout<NameElement->FirstChild-;
cout<AgeElement-<IDAttribute-}
int main{
string fileName"info.xml";
CreateXmlFile(fileName;
ReadXmlFile}

官方下载地址:http://www.oschina.net/p/tinyxml

本地下载地址:http://blogimg.chinaunix.net/blog/upfile2/100822222318.zip

在TinyXML中,根据XML的各种元素来定义了一些类:
TiXmlBase:整个TinyXML模型的基类。
TiXmlAttribute:对应于XML中的元素的属性。
TiXmlNode:对应于DOM结构中的节点。
TiXmlComment:对应于XML中的注释。
TiXmlDeclaration:对应于XML中的申明部分,即<?versiong="1.0" ?>。
TiXmlDocument:对应于XML的整个文档。
TiXmlElement:对应于XML的元素。
TiXmlText:对应于XML的文字部分。
TiXmlUnknown:对应于XML的未知部分。
TiXmlHandler:定义了针对XML的一些操作。

例如:

?xmlversion"1.0"standalone=no<!– Our to do list data –<ToDo<Item priority>Go to the<bold>Toy store!/bold>/Item>Do bills/ToDo>


整个对象树:

TiXmlDocument "demo.xml"
TiXmlDeclaration "version=’1.0′" "standalone=no"
TiXmlComment " Our to do list data"
TiXmlElement "ToDo"
TiXmlElement "Item" Attribtutes: priority = 1
TiXmlText "Go to the "
TiXmlElement "bold"
TiXmlText "Toy store!"
TiXmlElement "Item" Attributes: priority=2
TiXmlText "Do bills"

在tinyXML中,用FirstChild("名字")查找节点时,调用FirstChild函数的节点与要查找的节点必须成“父子关系”。

句柄

想要健壮地读取一个XML文档,检查方法调用后的返回值是否为null是很重要的。一种安全的检错实现可能会产生像这样的代码:


TiXmlElement*root=document.FirstChildElement"Document"if(root{
TiXmlElement*element=root"Element"(element*child=element"Child"(child*child2=child>NextSiblingElement(child2{
// Finally do something useful.

用句柄的话就不会这么冗长了,使用TiXmlHandle类,前面的代码就会变成这样:


TiXmlHandle docHandle&document=docHandle.FirstChild.Child.ToElement// do something useful


一、读取XML,设置节点文本
如下XML片段:

"1.0"encoding"UTF-8"standalone"yes"?<ZXML<ZAPP<VBS_RUNTIME_PARAMS<BROADCAST_VERSION info"版本">8/BROADCAST_VERSION<Broadcast<FileCount info"资源文件个数">69/FileCount<SOURCE_1<ID info"图片编号">1/ID<Versioninfo"图片版本"/Version<Path info"图片路径"/mnt/share/1.bmp/Path<FileMode info"文件处理模式">0/FileMode/SOURCE_1<SOURCE_2/2>2/SOURCE_2.
/Broadcast/VBS_RUNTIME_PARAMS/ZAPP/ZXML>

要设置BROADCAST_VERSION节点的值 8为其他值,可参考如下代码(将值加1):
用ReplaceChild( TiXmlNode* replaceThis,const TiXmlNode& withThis )方法替换


TiXmlDocument doc"zapp.conf";
doc.LoadFile;
TiXmlHandle docHandle&doc*Broadcast_ver"ZXML""ZAPP""VBS_RUNTIME_PARAMS""BROADCAST_VERSION";
TiXmlNode*oldnode=Broadcast_verconstchar*ver>GetTextintoldVeratoi(ver;
CString newVer;
newVer.Format"%d"+1;
TiXmlText newText(newVer;
Broadcast_ver>ReplaceChild(oldnode;
AfxMessageBox(Broadcast_ver;//输出值

doc.SaveFile;

二,删除节点,属性值

RemoveChild( TiXmlNode* removeThis )方法删除父节点的子节点,
RemoveAttribute( const char * name )方法删除属性值.

例如删除BROADCAST_VERSION节点

TiXmlHandledocHandle( & doc);
TiXmlElement
* Broadcast_ver = docHandle.FirstChild( " ZXML ).FirstChild( ZAPP VBS_RUNTIME_PARAMS ).ToElement();

TiXmlNode
node -> FirstChild( BROADCAST_VERSION );

Broadcast_ver
RemoveChild(node);

也可以删除整个SOURCE_1节点:

Broadcast SOURCE_1 );

Broadcast
RemoveChild(node);

删除BROADCAST_VERSION的info属性:

).FirstChildElement( ).ToElement();

Broadcast_ver
RemoveAttribute( info ); // 删除info

可以借助NextSiblingElement()方法实现递归删除.

三,添加节点,属性值

例如在SOURCE_3下添加BROADCAST_PID节点:

).ToElement();
TiXmlElement
Broadcast_Pid new TiXmlElement( BROADCAST_PID );
TiXmlText
text TiXmlText( 7215 );
Broadcast_Pid
SetAttribute( , thepid LinkEndChild(text);
Broadcast
LinkEndChild(Broadcast_Pid);

将在SOURCE_3后添加新的节点:

< BROADCAST_PID ="thepid" > </ >

四,最后说一下中文乱码的问题

乱码是由于GB2312与UTF8之间转换不当造成的,tinyxml在处理UTF8本身没有问题,当你打开一个UTF8的文档,可以在加载的时候指定UTF8的方式,或者文档声明处指明的编码格式,tinyxml会按照相应的编码格式加载,但很多时候当我们输出或写入中文字段时会出现乱码,无论在内存,还是打印出来的内容.这是因为我们的软件通常是GB2312编码,而读取或写入的内容是UTF8,自然就会出错.可以借助网上的两个函数来实现转换(原作者不详):

voidConvertUtf8ToGBK(CString&strUtf8intlen=MultiByteToWideChar(CP_UTF8(LPCTSTR)strUtf8-1NULLunsignedshort*wszGBK=new[len;
memset(wszGBK*2+2;
MultiByteToWideChar;

len=WideCharToMultiByte(CP_ACP*szGBK(szGBK;
WideCharToMultiByte;

strUtf8=szGBKdelete[]szGBK]wszGBK}


voidConvertGBKToUtf8&strGBK)strGBK*wszUtf8(wszUtf8*szUtf8(szUtf8;

strGBK=szUtf8]szUtf8]wszUtf8}

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读