- 浏览: 2035682 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (651)
- ACE (35)
- BAT (9)
- C/C++ (116)
- fast-cgi (14)
- COM (27)
- python (59)
- CGI (4)
- C# (2)
- VC (84)
- DataBase (29)
- Linux (96)
- P2P (6)
- PHP (15)
- Web (6)
- Memcached (7)
- IME输入法 (11)
- 设计模式 (2)
- 搜索引擎 (1)
- 个人情感 (4)
- 笔试/面试 (3)
- 一亩三分地 (33)
- 历史 (2)
- 地理 (1)
- 人物 (3)
- 经济 (0)
- 不仅仅是笑哦 (43)
- 小故事大道理 (2)
- http://www.bjdsmyysjk120.com/ (0)
- http://www.bjdsmyy120.com/ (0)
- 它山之石可以攻玉 (15)
- 大学生你关注些什么 (28)
- 数据恢复 (1)
最新评论
-
luokaichuang:
这个规范里还是没有让我明白当浏览器上传文件时,STDIN的消息 ...
FastCGI规范 -
effort_fan:
好文章!学习了,谢谢分享!
com技术简介 -
vcell:
有错误os.walk(strPath)返回的已经是全部的文件和 ...
通过python获取目录的大小 -
feifeigd:
feifeigd 写道注意:文章中的CPP示例第二行 #inc ...
ATL入门:利用ATL编写简单的COM组件 -
feifeigd:
注意:文章中的CPP示例第二行 #include " ...
ATL入门:利用ATL编写简单的COM组件
实战TinyXML
作者:裕作
(本文为原创,转贴请注明出处:http://blog.csdn.net/KyosukeNo1)
这几天在埋头写自己的3D文件浏览器(稍后发布),突发奇想的要把自己的内部格式转化成XML,于是,把以前在研究所时用过的ExPat翻了出来。 ExPat是基于事件的XML解释器,速度挺快的,但结构方面有点不敢恭维--当年写配置文件的导出导入部分花了我足足1个星期!而且由于它是基于事件发生的次序(SAX),似乎有时会发生一些无法控制的情况--例如进入某Level后忘了记录,结果……后面的程序全部死掉!这时想起同事之前推荐的 TinyXML,结果……用了不到3小时就把我的文件导出来了~~呵呵。在阅读本文之前,请先看看我Blog里转贴的《TinyXML学习笔记》,相信它能给各位一个关于TinyXML的初步概念。
言归正传,本文目的在于补全之前《TinyXML学习》的不足,尽量把常用的示例代码列出让大家参考。此外,在本篇最后会给出一个完整的文件读写例子,供读者参考。
1. 编程环境的设置。新建一个项目,起名叫TestTXML。到http://sourceforge.net/projects/tinyxml/ 下载TinyXML的官方例子,并编译第一个Project tinyxml(注意,最好编译Release的版本,代码比较小。然后把生成的tinyxml.lib(如果是Debug版本,叫tinyxmld.lib)连同tinystr.h和tinyxml.h一起Copy到TestTXML项目的目录中。在TestTXML项目里的头文件加入对TinyXML的引用:
#pragma comment(lib,"tinyxml.lib") // 链接Library
#include "tinyxml.h" // TinyXML的头文件
2. 建立一个XML文件:
char* sFilePath = "ikk_doc.xml"; // 文件名称
TiXmlDocument xmlDoc( sFilePath ); // 建立一个XML文件
TiXmlDeclaration Declaration( "1.0","gb2312", "yes" ); // 声明XML的属性
xmlDoc.InsertEndChild( Declaration ); // 写入基本的XML头结构
xmlDoc.SaveFile(); // 把XML文件写入硬盘
这时,在硬盘上的TestXML项目目录里,ikk_doc.xml文件已经被创建出来了。
3. 在XML文件里插入Element
所谓的Element,就是在XML里面的Tag,例如在<resume name=”裕作”>简历内容</resume>中,“Resume”就是Element的名字,上述的整个字符串就是一个Element。在TinyXML里,插入Element的步骤如下:
TiXmlElement* pElm = NULL;
pElm = new TiXmlElement( "resumes" ); //定义当前的子节点 pElmParent.InsertEndChild( *pElm ); // 把子节点插入父节点中
4. 在element里插入属性。在刚才例子中,name=”裕作”就是Resume的属性,其中name是属性的名字,”裕作”是属性的值。在当前子节点内插入属性的方法如下:
pElm->SetAttribute( "name", resume.sName );
5. 在XML里插入文本。在<resume name=”裕作”>简历内容</resume>中,“简历内容”就是一段文本,事实上,在TinyXML里,它是被当作一个Text类型的子节点来插入的。还而言之,就是在Resume的子节点中,插入这个Text子节点。插入例子如下:
TiXmlText* pText = NULL;
pText = new TiXmlText( "简历内容" ); // 定义文本的内容
pElmChild->InsertEndChild( *pText ); //把text子节点插入父节点中
在具备了以上背景知识之后,我们已经可以用TinyXML读写一个XML文件了。本文最后的程序将写入,然后重新读取一个XML文件到我们的结构里。这个XML文件的内容如下:
<?xml version="1.0" encoding="GB2312" ?>
<resumes>
<resume name="裕作">
<gender>男</gender>
<age>26</age>
<skills num="2">
<skill level="99">编程</skill>
<skill level="1">吹牛</skill>
</skills>
</resume>
<resume name="裕作 The Great">
<gender>男</gender>
<age>0</age>
<skills num="1">
<skill level="100">编程</skill>
</skills>
</resume>
</resumes>
以下程序将建立ikk_doc.xml文件,然后重新把内容读取进内存:
#pragma comment(lib,"tinyxml.lib")
#include "string.h"
#include "stdio.h"
#include "tinyxml.h"
#define XML_FILE "ikk_doc.xml"
#define NAME_LENGTH 256 // 名字类字符的分配长度
#define SAFE_DELETE(x) {if(x) delete x; x=NULL;} // 安全删除new分配出来的变量空间
#define SAFE_DELETE_ARRAY(x) {if(x) delete[] x; x=NULL;} // 安全删除new分配出来的数组空间
#define XML_HEADER "<?xml version=\"1.0\" encoding=\"GB2312\" ?>" // XML文件头的定义
typedef unsigned int uint32;
// 技能的结构
typedef struct skill_s {
uint32 nLevel; // 技能的程度
char sName[ NAME_LENGTH ]; // 技能的名称
skill_s() {
nLevel = 0;
sName[0] = 0;
}
} skill_t;
// 简历的结构
typedef struct resume_s {
char sName[ NAME_LENGTH ]; // 名字
bool isMan; // 是否男性
uint32 nAge; // 年龄
uint32 nNumSkill; // 技能的数目
skill_t* pSkill; // 技能的结构
resume_s() {
sName[0] = 0;
isMan = false;
nAge = 0;
nNumSkill = 0;
pSkill = NULL;
}
} resume_t;
void exportSkill( TiXmlElement* pElmParent, skill_t skill )
{
int i;
char sBuf[NAME_LENGTH]; // 一个临时存放的字符串
TiXmlElement* pElm = NULL; // 一个指向Element的指针
TiXmlText* pText = NULL; // 一个指向Text的指针
pElm = new TiXmlElement( "skill" );
// 插入等级(以属性形式)
sprintf( sBuf, "%d", skill.nLevel ); // 把Skill的登记变成字符串临时存进sBuf里
pElm->SetAttribute( "level", sBuf ); // 把等级插入Skill里
// 插入技能名称(以子Element形式)
pText = new TiXmlText( skill.sName ); // 建立一个Skill的子Element(一个Text形式的子元素)
pElm->InsertEndChild( *pText ); // 把这个Skill的子Element插入Skill里
SAFE_DELETE( pText ); // 删除这个Text
// 最后把整个Resume的子节点插入到父节点中
pElmParent->InsertEndChild( *pElm );
}
void importSkill( TiXmlElement* pElm, skill_t* pSkill )
{
int i;
char sBuf[NAME_LENGTH]; // 一个临时存放的字符串
TiXmlElement* pElmChild = NULL; // 一个指向Element的指针
TiXmlText* pText = NULL; // 一个指向Text的指针
// 读取level
pSkill->nLevel = atoi( pElm->Attribute( "level" ) );
// 读取技能名称
strcpy( pSkill->sName, pElm->FirstChild()->Value() );
}
void exportResume( TiXmlElement* pElmParent, resume_t resume )
{
int i;
char sBuf[NAME_LENGTH]; // 一个临时存放的字符串
TiXmlElement* pElm = NULL; // 一个指向Element的指针
TiXmlElement* pElmChild = NULL; // 一个指向Element的指针
TiXmlText* pText = NULL; // 一个指向Text的指针
pElm = new TiXmlElement( "resume" );
// 插入名字(以属性形式)
pElm->SetAttribute( "name", resume.sName );
// 插入性别(以子Element形式)
pElmChild = new TiXmlElement( "gender" ); // 建立一个子Element叫Gender
if( resume.isMan )
pText = new TiXmlText( "男" ); // 建立一个Gender的子Element(一个Text形式的子元素)
else
pText = new TiXmlText( "女" ); // 建立一个Gender的子Element(一个Text形式的子元素)
pElmChild->InsertEndChild( *pText ); // 把这个Gender的子Element插入Gender里
pElm->InsertEndChild( *pElmChild ); // 把Gender插入到主Element里
SAFE_DELETE( pElmChild ); // 删除已经用完的Gender
SAFE_DELETE( pText ); // 删除这个Text
// 插入年龄(以子Element形式)
pElmChild = new TiXmlElement( "age" ); // 建立一个子Element叫Age
sprintf( sBuf, "%d", resume.nAge ); // 把Age变成字符串临时存进sBuf里
pText = new TiXmlText( sBuf ); // 建立一个Age的子Element(一个Text形式的子元素)
pElmChild->InsertEndChild( *pText ); // 把这个Age的子Element插入Age里
pElm->InsertEndChild( *pElmChild ); // 把Age插入到主Element里
SAFE_DELETE( pElmChild ); // 删除已经用完的Age
SAFE_DELETE( pText ); // 删除这个Text
// 插入技能子节点
pElmChild = new TiXmlElement( "skills" ); // 建立一个子Element叫Skills
sprintf( sBuf, "%d", resume.nNumSkill ); // 把Skill的数目变成字符串临时存进sBuf里
pElmChild->SetAttribute( "num", sBuf ); // 把这个Skills的属性插入Skills里
for( i=0; i<resume.nNumSkill; i++ )
{
exportSkill( pElmChild, resume.pSkill[i] ); // 插入一项技能
}
pElm->InsertEndChild( *pElmChild ); // 把Skills插入到主Element里
SAFE_DELETE( pElmChild ); // 删除已经用完的Skills
SAFE_DELETE( pText ); // 删除这个Text
// 最后把整个Resume的子节点插入到父节点中
pElmParent->InsertEndChild( *pElm );
SAFE_DELETE( pElm ); // 删除子节点
}
void importResume( TiXmlElement* pElm, resume_t* pResume )
{
int i;
char sBuf[NAME_LENGTH]; // 一个临时存放的字符串
TiXmlElement* pElmChild = NULL; // 一个指向Element的指针
TiXmlElement* pElmGrandChild = NULL; // 一个指向Element的指针
TiXmlText* pText = NULL; // 一个指向Text的指针
// 读入"resume"子节点
strcpy( pResume->sName, pElm->Attribute( "name" ) );
// 读入"gender"子节点
pElmChild = pElm->FirstChildElement( "gender" );
if( strcmp( "男", pElmChild->FirstChild()->Value() ) == 0 )
pResume->isMan = true;
else
pResume->isMan = false;
// 读入"age"子节点
pElmChild = pElm->FirstChildElement( "age" );
pResume->nAge = atoi( pElmChild->FirstChild()->Value() );
// 读入"skills"子节点
pElmChild = pElm->FirstChildElement( "skills" );
pResume->nNumSkill = atoi( pElmChild->Attribute( "num" ) );
pResume->pSkill = new skill_t[pResume->nNumSkill];
pElmGrandChild = pElmChild->FirstChildElement( "skill" ); // 指向第一个Skill
for( i=0; i<pResume->nNumSkill; i++ ) {
importSkill( pElmGrandChild, &(pResume->pSkill[i]) ); // 读取一个Skill
pElmGrandChild = pElmGrandChild->NextSiblingElement(); // 指向下一个Skill
}
}
bool readXML( char* sFilePath, int* nNumResume, resume_t** ppResume ) {
int i; // 用做循环的变量
TiXmlElement* pElmChild = NULL; // 一个指向Element的指针
TiXmlDocument xmlDoc( sFilePath ); // 输入XML路径
if( !xmlDoc.LoadFile() ) // 读取XML并检查是否读入正确
return false;
TiXmlElement* pElmRoot = NULL; // 根节点
pElmRoot = xmlDoc.FirstChildElement( "resumes" ); // 得到根节点
if( !pElmRoot ) {
return false;
}
*nNumResume = atoi( pElmRoot->Attribute( "num" ) ); // 读取Resume的数目
*ppResume = new resume_t[*nNumResume]; // 分配Resume的空间
pElmChild = pElmRoot->FirstChildElement( "resume" ); // 找出第一个Resume
for( i=0; i<*nNumResume; i++ ) {
importResume( pElmChild, &((*ppResume)[i]) ); // 读取Resume的内容
pElmChild = pElmChild->NextSiblingElement(); // 找出下一个Resume
}
return true;
}
bool writeXML( char* sFilePath, int nNumResume, resume_t* pResume )
{
if( !sFilePath || !pResume )
return false; // 确定指针存在
int i; // 用做循环的变量
char sBuf[NAME_LENGTH]; // 一个临时存放的字符串
TiXmlElement* pElm = NULL; // 一个指向Element的指针
TiXmlDeclaration Declaration( "1.0","gb2312", "yes" ); // 建立XML头结构
TiXmlDocument xmlDoc( sFilePath ); // 用存档的文件名字来建立一个XML文件
xmlDoc.InsertEndChild( Declaration ); // 把XML头结构插入当前文档
// 插入根节点“Resumes”
pElm = new TiXmlElement( "resumes" ); // 建立根节点“Resumes”
sprintf( sBuf, "%d", nNumResume ); // 把nNumResume变成字符串临时存进sBuf里
pElm->SetAttribute( "num", sBuf ); // 建立一个Resumes的子Element
for( i=0; i<2; i++ )
{
exportResume( pElm, pResume[i] ); // 在根节点上插入以上定义的2个简历
}
xmlDoc.InsertEndChild( *pElm );
xmlDoc.SaveFile();
SAFE_DELETE( pElm ); // 删除Element
return true;
}
void main()
{
int i, j;
// + == 设置两份简历 ==========================================================
int nNumResume = 2;
resume_t* pResume = new resume_t[ nNumResume ];
// 1. 初始化第一份简历
strcpy( pResume[0].sName, "裕作" );
pResume[0].isMan = true;
pResume[0].nAge = 26;
pResume[0].nNumSkill = 2;
pResume[0].pSkill = new skill_t[2];
{ // 设置技能列表结构
strcpy( pResume[0].pSkill[0].sName, "编程" );
strcpy( pResume[0].pSkill[1].sName, "吹牛" );
pResume[0].pSkill[0].nLevel = 99;
pResume[0].pSkill[1].nLevel = 1;
}
// 2. 初始化第二份简历
strcpy( pResume[1].sName, "裕作 The Great" );
pResume[1].isMan = true;
pResume[1].nAge = 0;
pResume[1].nNumSkill = 1;
pResume[1].pSkill = new skill_t[1];
{ // 设置技能列表结构
strcpy( pResume[1].pSkill[0].sName, "编程" );
pResume[1].pSkill[0].nLevel = 100;
}
// - == 设置两份简历 ==========================================================
// 把简历以XML形式写入磁盘
if( !writeXML( XML_FILE, nNumResume, pResume ) )
{
printf( "ERROR: can't write the file." );
return;
}
// 删除Resume
nNumResume = 0;
SAFE_DELETE_ARRAY( pResume );
// 重新读入XML文件里的Resume数据
if( !readXML( XML_FILE, &nNumResume, &pResume ) )
{
printf( "ERROR: can't read the file." );
return;
}
// 把所有简历输出到屏幕
if( pResume ) // 确定有Resume
{
for( i=0; i<nNumResume; i++ ) {
printf( "简历:======================\n" );
printf( "\t名字:%s\n", pResume[i].sName );
if( pResume[i].isMan )
printf( "\t性别:男\n" );
else
printf( "\t性别:女\n" );
printf( "\t年龄:%d\n", pResume[i].nAge );
printf( "\t职业技能:\n" );
for( j=0; j<pResume[i].nNumSkill; j++ ) {
printf( "\t\t技能名称:%s\n", pResume[i].pSkill[j].sName );
printf( "\t\t技能等级:%d\n", pResume[i].pSkill[j].nLevel );
}
}
}
}
发表评论
-
Berkeley DB 使用经验总结
2012-08-27 14:41 3086作者:陈磊 NoSQL是现在互联网Web2.0时代备受 ... -
嵌入式数据库系统Berkeley DB
2012-08-27 14:37 1533前言 UNIX/LINUX平台下的数据库种类非常多 ... -
C语言中标准输入流、标准输出流、标准错误输出流
2011-06-13 14:32 9291C语言中标准输入流、标准输出流、标准错误输出流 在 ... -
Rsync 实现原理
2011-05-12 20:06 8318Rsync 实现原理 前言 关于rsync的原始文档 ... -
c++简单的虚函数测试
2011-04-27 14:25 1019#include <iostream> u ... -
C++文件行查找
2011-04-26 14:10 1408#include <iostream> # ... -
c++偏特化简单示例
2011-04-13 11:17 2156c++偏特化 // temp1.c ... -
GDB调试精粹及使用实例
2011-03-16 14:06 1140GDB调试精粹及使用实例 一:列文件清单 1. ... -
简单的ini文件解析
2011-02-12 16:36 1619int GetKeyVal(const string s ... -
scanf族函数高级用法
2011-01-25 16:00 2559如何解释 fscanf(fd,&quo ... -
使用scons替代makefile(1)
2011-01-25 11:58 3730早在多年前我刚开始接触linux下的C程序时,经常被makef ... -
使用scons替代makefile(2)
2011-01-25 11:57 3583本篇文章接着上一篇进一步介绍scons的使用方法,主要介绍静态 ... -
使用scons替代makefile(3)
2011-01-25 11:55 4821在上两篇文章中已经简单介绍了用scons编译库文件,可执行程序 ... -
C 支持动态添加测试数据的测试代码
2011-01-13 17:22 1119/下面的定义为了支持可扩增。 //当需要增加一个新的测试用列 ... -
Linux下Makefile的automake生成
2010-12-28 16:55 1101******************helloworld.c* ... -
SCons笔记(详细版)
2010-12-23 16:11 105361. 基本使用 SConstruct文件就功能而言相当于Ma ... -
scons 学习
2010-12-23 11:14 2181scons 学习 作者:Sam(甄峰) sam_code@h ... -
scons随笔
2010-12-22 20:20 4706scons随笔 Scons是新一代的软件构件工具,或者说ma ... -
Scons在linux下的安装和使用
2010-12-21 11:59 3285因为正在用的一个开源软件需要的Developm ... -
排列组合的实现
2010-12-20 12:41 1063简单算法: 从前往后(或者从后往前)每次交换一个位置。当存在 ...
相关推荐
10. **实战应用**:在实际项目中,Tinyxml2常用于配置文件的读取、游戏数据的存储、软件的序列化等方面。 通过分析和运行压缩包中的示例代码,你可以更深入地理解Tinyxml2库的用法,并将其应用于自己的C++项目中。...
【标题】"金山开源BKUI.16.Tinyxml"是一个关于金山软件开源项目中使用的Tinyxml库的源代码集合。Tinyxml是一个轻量级的XML解析器,它旨在简化XML文档的读写操作,尤其适合嵌入式系统或资源有限的环境中。金山软件在...
- C++:可以使用标准模板库(STL)中的XML库,如pugixml或TinyXML,实现XML操作。 - C#:.NET的C#编程中,XML处理更为便捷,如LINQ to XML提供了一种更简洁的API来创建和查询XML文档。 - Java:JDK提供了javax....
1. tinyxml.cpp、tinyxmlparser.cpp、tinystr.cpp 和 tinyxmlerror.cpp:这些是 TinyXML 库的源代码,这是一个轻量级的 XML 解析库,用于读写 XML 文件。在这个示例中,它们可能被用作测试的基础,或者是为了演示...
2. tinyxml:同样是C++的XML库,小巧且功能齐全,支持DOM和SAX解析,适用于嵌入式系统和小型应用。 3. libxml2:由GNU项目开发,支持DOM、SAX和XPath,功能强大,广泛应用于各种平台。 4. lxml(Python):结合了...
实现注册、注销、心跳、实时流、历史流,将开发期间的所有下载库都压缩在一起,包括BOOST、ffmpeg、alsa、SDL2、tinyxml_2_6_2、yasm,便于直接进行开发编译,希望可以跟大家一起交流学习
数据集通常包含xml文件和jpg图像,xml文件用于标注每个图像中的火焰位置和大小,jpg文件则为原始图像。在训练模型前,需要先对这些数据进行预处理,包括图片的缩放、归一化以及标注信息的转换,以便于模型学习。 4...
- **XML与C++结合**:使用库如pugixml或TinyXML处理XML文件,进行数据存储和交换。 5. **实战项目**: - **小程序开发**:通过编写简单的程序,如计算器、文本编辑器等,将理论知识付诸实践。 - **游戏编程**:...
5. 数据解析:掌握XML或JSON数据的解析,这通常涉及到第三方库,如TinyXML或pugixml对于XML,或者jsoncpp或nlohmann/json对于JSON。 6. 异步编程:飞信的网络操作通常是异步的,需要理解异步编程模型,如消息队列、...
为了解析XML,我们可以使用内置的`XmlPullParser`或者第三方库如`TinyXML`、`Jackson`或`Gson`,这些库能帮助我们更方便地处理XML数据。 在UI设计方面,Android提供了丰富的布局组件,如`RecyclerView`可以用来显示...
7. **HTML解析**:可能使用C++库如pugixml或TinyXML来解析和操作HTML文件。 8. **网络编程**:如果涉及HTTP通信,可能涉及到套接字编程或第三方库如libcurl。 9. **单元测试**:使用如Google Test这样的测试框架...
【WeQuest - 简单微型聊天式系统 开源详解】 WeQuest是一个开源的轻量级CRM(Customer Relationship ...通过深入研究和使用WeQuest,不仅可以掌握PHP和AJAX的实战应用,还能了解CRM系统的基本构建和运作原理。
文档中提到的“1000行代码读懂Spring核心”,指的是通过一个名为tiny-spring的开源项目,来逐步讲解Spring框架的实现原理。这个项目是对Spring框架核心功能的一个简化版本,通过逐行代码的分析,来帮助开发者理解...
TBS还提供了其他高级特性,如自定义函数、插件扩展、XML和CSV处理、邮件发送等。例如,通过`TBS->Plugin(OP_TBS_METHOD, params)`可以调用内置或自定义的插件。 8. 故障排查: 在开发过程中,如果遇到问题,TBS提供...
书中可能还会讲解如何配置Struts2的XML配置文件,以及如何使用注解进行简化配置。此外,可能会涵盖Struts2与Spring、Hibernate等其他框架的整合,以及Struts2的国际化和本地化支持。 2. **《STRUTS2整理成册517》**...
7. **网络编程**:ActionScript3.0支持XML、JSON等数据格式的处理,以及Socket、URLLoader等类用于网络通信,使得可以创建与服务器交互的动态应用。 8. **音频和视频处理**:教程可能涵盖如何加载、播放、控制和...