TiXmlHandle的使用-简化tinyxml的代码

借鉴于:http://www.cppblog.com/elva/archive/2008/04/24/47907.html

例如:

1 <?xml version="1.0" standalone=no>
2 <!– Our to do list data –>
3 <ToDo>
4 <Item priority="1"> Go to the <bold>Toy store!</bold></Item>
5 <Item priority="2"> Do bills</Item>
6 </ToDo> 

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

 1 TiXmlElement* root = document.FirstChildElement( "Document" );
 2 if ( root )
 3 {
 4 TiXmlElement* element = root->FirstChildElement( "Element" );
 5 if ( element )
 6 {
 7 TiXmlElement* child = element->FirstChildElement( "Child" );
 8 if ( child )
 9 {
10 TiXmlElement* child2 = child->NextSiblingElement( "Child" );
11 if ( child2 )
12 {
13 // Finally do something useful. 

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

1 TiXmlHandle docHandle( &document );
2 TiXmlElement* child2 = docHandle.FirstChild( "Document" ).FirstChild( "Element" ).Child( "Child", 1 ).ToElement();
3 if ( child2 )
4 {
5 // do something useful 

一、读取XML,设置节点文本

如下XML片段:

 1 <?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
 2 <ZXML>
 3     <ZAPP>
 4          <VBS_RUNTIME_PARAMS>
 5                <BROADCAST_VERSION info="版本">8</BROADCAST_VERSION>
 6                <Broadcast>
 7                 <FileCount info="资源文件个数">69</FileCount>
 8                 <SOURCE_1>
 9                     <ID info="图片编号">1</ID>
10                     <Version info="图片版本">1</Version>
11                     <Path info="图片路径">/mnt/share/1.bmp</Path>
12                     <FileMode info="文件处理模式">0</FileMode>
13                 </SOURCE_1>
14                 <SOURCE_2>
15                     <Path info="图片路径">/mnt/share/2.bmp</Path>
16                     <ID info="图片编号">2</ID>
17                     <Version info="图片版本">1</Version>
18                     <FileMode info="文件处理模式">0</FileMode>
19                 </SOURCE_2>
20                 .
21                 </Broadcast>
22          </VBS_RUNTIME_PARAMS>
23     </ZAPP>
24 </ZXML>

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

 1 TiXmlDocument doc("zapp.conf");
 2     doc.LoadFile();
 3     TiXmlHandle docHandle( &doc );
 4 TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChildElement("BROADCAST_VERSION").ToElement();
 5     TiXmlNode * oldnode =  Broadcast_ver->FirstChild();
 6     const char *ver = Broadcast_ver->GetText();
 7     int oldVer = atoi(ver);
 8     CString newVer;
 9     newVer.Format("%d",oldVer+1);
10     TiXmlText newText(newVer);
11     Broadcast_ver->ReplaceChild(oldnode,newText);
12     AfxMessageBox(Broadcast_ver->GetText());//输出值
13     doc.SaveFile();

二,删除节点,属性值

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

例如删除BROADCAST_VERSION节点

1 TiXmlHandle docHandle( &doc );
2     TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").ToElement();
3
4     TiXmlNode * node =  Broadcast_ver->FirstChild("BROADCAST_VERSION");
5
6     Broadcast_ver->RemoveChild(node);

也可以删除整个SOURCE_1节点:

1 TiXmlHandle docHandle( &doc );
2     TiXmlElement* Broadcast = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChild("Broadcast").ToElement();
3
4     TiXmlNode * node =  Broadcast->FirstChild("SOURCE_1");
5
6     Broadcast->RemoveChild(node);

删除BROADCAST_VERSION的info属性:

1 TiXmlHandle docHandle( &doc );
2     TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChildElement("BROADCAST_VERSION").ToElement();
3
4     Broadcast_ver->RemoveAttribute("info"); //删除info

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

三,添加节点,属性值

例如在SOURCE_3下添加BROADCAST_PID节点:

1 TiXmlHandle docHandle( &doc );
2     TiXmlElement* Broadcast = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChild("Broadcast").ToElement();
3     TiXmlElement* Broadcast_Pid = new TiXmlElement("BROADCAST_PID");
4     TiXmlText *text =new TiXmlText("7215");
5     Broadcast_Pid->SetAttribute("info","the pid");
6     Broadcast_Pid->LinkEndChild(text);
7     Broadcast->LinkEndChild(Broadcast_Pid);

将在SOURCE_3后添加新的节点:

1 <BROADCAST_PID info="the pid">7215</BROADCAST_PID>

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

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

 1 void ConvertUtf8ToGBK(CString& strUtf8)
 2     {
 3         int len=MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, NULL,0);
 4         unsigned short * wszGBK = new unsigned short[len+1];
 5         memset(wszGBK, 0, len * 2 + 2);
 6         MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, wszGBK, len);
 7
 8         len = WideCharToMultiByte(CP_ACP, 0, wszGBK, -1, NULL, 0, NULL, NULL);
 9         char *szGBK=new char[len + 1];
10         memset(szGBK, 0, len + 1);
11         WideCharToMultiByte (CP_ACP, 0, wszGBK, -1, szGBK, len, NULL,NULL);
12
13         strUtf8 = szGBK;
14         delete[] szGBK;
15         delete[] wszGBK;
16     }
17
18
19     void ConvertGBKToUtf8(CString& strGBK)
20     {
21         int len=MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, NULL,0);
22         unsigned short * wszUtf8 = new unsigned short[len+1];
23         memset(wszUtf8, 0, len * 2 + 2);
24         MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, wszUtf8, len);
25
26         len = WideCharToMultiByte(CP_UTF8, 0, wszUtf8, -1, NULL, 0, NULL, NULL);
27         char *szUtf8=new char[len + 1];
28         memset(szUtf8, 0, len + 1);
29         WideCharToMultiByte (CP_UTF8, 0, wszUtf8, -1, szUtf8, len, NULL,NULL);
30
31         strGBK = szUtf8;
32         delete[] szUtf8;
33         delete[] wszUtf8;
34     }

当然,你也可以用MultiByteToWideChar,WideCharToMultiByte函数自己实现转换.以上是简单应用的几个举例,理解他们,相信你已经能写出满足自己需要的代码了.

时间: 2024-10-26 08:22:04

TiXmlHandle的使用-简化tinyxml的代码的相关文章

UWP: 妙用自定义 Action 以简化并重用代码

原文:UWP: 妙用自定义 Action 以简化并重用代码 相信每一位 App 开发者,在开发过程中,都会有一些代码被反复用到,比如:复制文本,打电话,发短信,发邮件,给应用添加评论等等.在项目之间复制这些代码段,实在不是一个好办法,所以大家可能会把这些代码放到一个类似 Utility 类中,或者一个库(Class Library)中,再加以调用.本文也是帮你完成同样的事情,不过本文是通过 Action 来完成.这两种方法的目的相同,区别是前者尽管实现了代码段封装,还是需要写代码(通常在 Vie

匿名内部类创建线程,简化线程创建代码

1 package cn.learn.thread.Thread; 2 /* 3 匿名内部类方式的实现线程的创建 4 5 匿名:没有名字 6 内部类:写在其他类的内部 7 8 作用:简化代码,不用单独写一个类,来设置线程任务 9 把子类继承父类,重写父类方法,创建子类对象合成一部完成 10 或者实现类实现接口,重写接口方法,创建实现类对象一部完成 11 匿名内部类的最终产物:子类/实现类对象,没有名字 12 13 格式: 14 new 父类/接口{ 15 重写父类/接口方法 16 } 17 18

UWP 开发: 妙用自定义 Action 以简化并重用代码

相信每一个 App 开发者,在开发过程中,都会有一些代码被反复用到,比如:复制文本,打电话,发短信,发邮件,给应用添加评论等等.在项目之间复制这些代码段,实在不是一个好办法,所以大家可能会把这些代码放到一个类似 Utility 类中,或者一个库(Class Library)中.本文也是帮你完成同样的事情,不过本文是通过 Action 来完成.这两种方法的目的相同,区别是前者尽管实现了代码段封装,还是需要写代码(通常在 ViewModel 中)来调用,而后者,则只要在 XAML 上添加几行代码即可

如何简化实体类代码

Spring boot使用Lombok编码 添加依赖 在 pom.xml 文件中添加相关依赖: <lombok.version>1.16.20</lombok.version> <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok --> <dependency> <groupId>org.projectlombok</groupId> <artifac

工作中遇到的问题--Hibernate一对多保存简化Service层代码

在创建一方的entity是添加一个增加多方的方法: package com.sim.mfg.data.domain; import java.io.Serializable;import java.util.Date;import java.util.HashMap;import java.util.Map;import java.util.Set; import javax.persistence.CascadeType;import javax.persistence.Column;impo

Map功能简化Python并发代码

<转摘>Python 并行任务技巧 支持Map并发的包文件有两个: Multiprocessing,还有少为人知的但却功能强大的子文件 multiprocessing.dummy. Dummy是一个多进程包的完整拷贝.唯一不同的是,多进程包使用进程,而dummy使用线程(自然也有Python本身的一些限制).所以一个有的另一个也有.这样在两种模式间切换就十分简单,并且在判断框架调用时使用的是IO还是CPU模式非常有帮助. 导入相关包 1 from multiprocessing import

使用Lombok简化你的代码

http://www.cnblogs.com/ywqbj/p/5711691.html mavem <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId></dependency> @data 注解在类上,相当于同时使用了@ToString.@EqualsAndHashCode.@Getter.@Setter和@Required

基于MVC+EasyUI的Web开发框架经验总结(11)--使用Bundles处理简化页面代码

在Web开发的时候,我们很多时候,需要引用很多CSS文件.JS文件,随着使用更多的插件或者独立样式文件,可能我们的Web界面代码会越来越臃肿,看起来也很累赘,在MVC里面提供了一个Bundle的对象,用来简化页面代码非常方便,本文主要介绍在我的MVC框架里面,如何使用bundles来简化页面的代码的. 1.常规的页面代码 我们知道,随着使用更多的一些效果,我们可能不断引入一些新的JS和CSS文件,已达到Web界面更好的表现效果.这样也就逐步增加了文件代码的行数,造成相对比较臃肿的场景,如下面的我

基于MVC4+EasyUI的Web开发框架经验总结(11)--使用Bundles处理简化页面代码

在Web开发的时候,我们很多时候,需要引用很多CSS文件.JS文件,随着使用更多的插件或者独立样式文件,可能我们的Web界面代码会越来越臃肿,看起来也很累赘,在MVC里面提供了一个Bundle的对象,用来简化页面代码非常方便,本文主要介绍在我的MVC框架里面,如何使用bundles来简化页面的代码的. 1.常规的页面代码 我们知道,随着使用更多的一些效果,我们可能不断引入一些新的JS和CSS文件,已达到Web界面更好的表现效果.这样也就逐步增加了文件代码的行数,造成相对比较臃肿的场景,如下面的我