TinyXML

TinyXml入门简易教程.

LG_Ting 2017-04-29 17:58:46 6172 收藏 6
分类专栏: 拓展库研究
版权
XML概念:

设计为传输和存储数据,其焦点是数据的内容。允许创作者定义自己的标签和自己的文档结构。既然是XML文件,就必须符合XML的规范,这里就不在赘述,必须要有根节点.

Dom概念:

Dom中的核心概念就是节点。DOM在分析XML文档时,将组成XML文档的各个部分(元素,属性,文本,注释,处理指令等等)映射为一个对象,这个对象就叫节点。

在内存中,这些节点形成一颗文档树。整棵树是一个节点,树中的每一个节点也是一棵树(子树),可以说,DOM就是对这棵树的一个对象描述,我们通过访问树中的

节点来存取XML文档的内容.

基于TinyXML对XML读写:重点在后面总结的类,熟悉并使用就好.

TinyXML开发无非就是对XML文档进行读写操作,网上开到的各种参考资料看的我头的大了,简而言之就是不合适.

一:读操作

假设有info.xml这个文件,然后将info.xml文件中的相关内容读出来.

info.xml

<PersonID="1">                   

    <nameID="100">周星星</name>

    <age>22</age>                

              <sex>男</sex>                     

</Person>                        

关键步骤解说:

/*

     写在前言:一些下面代码基本概念要点.

         1.       节点或者元素对象< TiXmlElement>: Personsnameagesex.其本身的value就是这些字符串.

         2.      若想获得节点标签内的值比如周星星.那就需要节点-> FirstChild()->value().

         3.      若想获取<PersonID="1">中这个ID的值,那么就节点->FirstAttribute()获得TiXmlAttribute *,然后再通过返回对象->Value()来获取返回值.

         这个地方本工程代码没有提及到,然后可能有多个属性,直接依照FirstAttribute的返回值对象调用Next()方法获得下一个对象.再通过这个对象取value即可.

         4.      要有相关层次概念和遍历的基本概念.比如获取<nameID="100">周星星</name>元素就是通过<Person ID="1">  元素调用FirstChildElement()获得,若想再同层次遍历则需要以<Person ID="1">为基准,不断渐进的NextSiblingElement()下去.

*/

     构建文档对象

     TiXmlDocument *myDocument =newTiXmlDocument(“文件名”).  



     将文件信息载入到文档对象中.

     myDocument->LoadFile();        

 

     依照文档对象找到根元素/根节点<Persons>

     TiXmlElement *RootElement =myDocument->RootElement();



     第二个<Person ID="1">是第一个<Persons>的子节点,通过FirstChildElement获得.

     TiXmlElement*FirstPerson =RootElement->FirstChildElement();

二:写操作

还是按照上面那个info.xml文件格式进行写xml文件.

info.xml

<PersonID="1">                   

    <nameID="100">周星星</name>

    <age>22</age>                

              <sex>男</sex>                     

</Person>                        

关键步骤解说:

     1.             其实写很简单,到时候仅需要用文档对象TiXmlDocument调用SaveFile即可完成写成一个xml文件了.关键在于如何构建TiXmlDocument文档对象以及如何对内容进行填充.

     2.             创建一个元素节点,并未元素阶段设置属性和内容

     创建节点并给节点value,也可以通过setvalue设定:

     TiXmlElement *NameElement =new TiXmlElement("name");



     设置节点标签中的内容:

     TiXmlText *NameContent =new TiXmlText("周星星");

     NameElement->LinkEndChild(NameContent);



     设置节点属性:

     NameElement->SetAttribute("ID","100");   

     NameElement->SetAttribute("benxin","200");



     3.             根元素挂载方式.

     TiXmlElement *RootElement=newTiXmlElement("Persons");

     myDocument->LinkEndChild(RootElement);



     4.             同一节点下挂载多个子节点

        PersonElement->LinkEndChild(NameElement);

        PersonElement->LinkEndChild(AgeElement);

唉,完全弄懂和编写这个文档好累啊,项目源码在这里<只需要一积分,可以的话权当下载支持我,谢谢>:http://download.csdn.net/detail/qq_24571549/9829253

下面是收藏别人的博客相关内容,拓展可以看他的.http://www.cppblog.com/elva/archive/2010/01/17/47907.html

在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的一些操作。

例如:

<?xml version="1.0" standalone=no> <!– Our to do list data –> Go to the Toy store! Do bills 整个对象树: 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->FirstChildElement( "Element" ); if ( element ) { TiXmlElement* child = element->FirstChildElement( "Child" ); if ( child ) { TiXmlElement* child2 = child->NextSiblingElement( "Child" ); if ( child2 ) { // Finally do something useful. 用句柄的话就不会这么冗长了,使用TiXmlHandle类,前面的代码就会变成这样: TiXmlHandle docHandle( &document ); TiXmlElement* child2 = docHandle.FirstChild( "Document" ).FirstChild( "Element" ).Child( "Child", 1 ).ToElement(); if ( child2 ) { // do something useful 一、读取XML,设置节点文本 如下XML片段: <?xml version="1.0" encoding="UTF-8" standalone="yes" ?> 8 69 1 1 /mnt/share/1.bmp 0 /mnt/share/2.bmp 2 1 0 .

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

TiXmlDocument doc("zapp.conf");
doc.LoadFile();
TiXmlHandle docHandle( &doc );

TiXmlElement* Broadcast_ver = docHandle.FirstChild(“ZXML”).FirstChild(“ZAPP”).FirstChild(“VBS_RUNTIME_PARAMS”).FirstChildElement(“BROADCAST_VERSION”).ToElement();
TiXmlNode * oldnode = Broadcast_ver->FirstChild();
const char *ver = Broadcast_ver->GetText();
int oldVer = atoi(ver);
CString newVer;
newVer.Format("%d",oldVer+1);
TiXmlText newText(newVer);
Broadcast_ver->ReplaceChild(oldnode,newText);
AfxMessageBox(Broadcast_ver->GetText());//输出值
doc.SaveFile();

二,删除节点,属性值

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

例如删除BROADCAST_VERSION节点

TiXmlHandle docHandle( &doc );
TiXmlElement* Broadcast_ver = docHandle.FirstChild(“ZXML”).FirstChild(“ZAPP”).FirstChild(“VBS_RUNTIME_PARAMS”).ToElement();

TiXmlNode * node =  Broadcast_ver->FirstChild("BROADCAST_VERSION");

Broadcast_ver->RemoveChild(node);

也可以删除整个SOURCE_1节点:

TiXmlHandle docHandle( &doc );
TiXmlElement* Broadcast = docHandle.FirstChild(“ZXML”).FirstChild(“ZAPP”).FirstChild(“VBS_RUNTIME_PARAMS”).FirstChild(“Broadcast”).ToElement();

TiXmlNode * node =  Broadcast->FirstChild("SOURCE_1");

Broadcast->RemoveChild(node);

删除BROADCAST_VERSION的info属性:

TiXmlHandle docHandle( &doc );
TiXmlElement* Broadcast_ver = docHandle.FirstChild(“ZXML”).FirstChild(“ZAPP”).FirstChild(“VBS_RUNTIME_PARAMS”).FirstChildElement(“BROADCAST_VERSION”).ToElement();

Broadcast_ver->RemoveAttribute("info"); //删除info

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

三,添加节点,属性值

例如在SOURCE_3下添加BROADCAST_PID节点:

TiXmlHandle docHandle( &doc );
TiXmlElement* Broadcast = docHandle.FirstChild(“ZXML”).FirstChild(“ZAPP”).FirstChild(“VBS_RUNTIME_PARAMS”).FirstChild(“Broadcast”).ToElement();
TiXmlElement* Broadcast_Pid = new TiXmlElement(“BROADCAST_PID”);
TiXmlText *text =new TiXmlText(“7215”);
Broadcast_Pid->SetAttribute(“info”,“the pid”);
Broadcast_Pid->LinkEndChild(text);
Broadcast->LinkEndChild(Broadcast_Pid);

将在SOURCE_3后添加新的节点:

<BROADCAST_PID info=“the pid”>7215</BROADCAST_PID>

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

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

void ConvertUtf8ToGBK(CString& strUtf8) 
{
    int len=MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, NULL,0);
    unsigned short * wszGBK = new unsigned short[len+1];
    memset(wszGBK, 0, len * 2 + 2);
    MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, wszGBK, len);

    len = WideCharToMultiByte(CP_ACP, 0, wszGBK, -1, NULL, 0, NULL, NULL);
    char *szGBK=new char[len + 1];
    memset(szGBK, 0, len + 1);
    WideCharToMultiByte (CP_ACP, 0, wszGBK, -1, szGBK, len, NULL,NULL);

    strUtf8 = szGBK;
    delete[] szGBK;
    delete[] wszGBK;
}


void ConvertGBKToUtf8(CString& strGBK)
{
    int len=MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, NULL,0);
    unsigned short * wszUtf8 = new unsigned short[len+1];
    memset(wszUtf8, 0, len * 2 + 2);
    MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, wszUtf8, len);

    len = WideCharToMultiByte(CP_UTF8, 0, wszUtf8, -1, NULL, 0, NULL, NULL);
    char *szUtf8=new char[len + 1];
    memset(szUtf8, 0, len + 1);
    WideCharToMultiByte (CP_UTF8, 0, wszUtf8, -1, szUtf8, len, NULL,NULL);

    strGBK = szUtf8;
    delete[] szUtf8;
    delete[] wszUtf8;
}

当然,你也可以用MultiByteToWideChar,WideCharToMultiByte函数自己实现转换.以上是简单应用的几个举例,理解他们,相信你已经能写出满足自己需要的
转载:https://blog.csdn.net/qq_24571549/article/details/70980125

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值