TinyXml2是开源的c++xml文件解析库,简单实用,非常适合存储简单数据,配置文件,对象序列化等数据量不是很大的操作。

  (1)DOM

DOM(Document Object Model),即文档对象模型,是针对XML并经过扩展用于HTML的应用程序编程接口(API)。

DOM本质上是一种接口(API),是专门操作网页内容的API标准。DOM把整个页面映射为一个多层节点结构,HTML或XML页面中的每个组成部分都是某种类型的节点。借助DOM提供的API,开发人员可以删除、添加、替换或修改任何节点。

DOM标准被分为三个不同的部分:
.核心 DOM - 针对任何结构化文档的标准模型
.XML DOM - 针对 XML 文档的标准模型
.HTML DOM - 针对 HTML 文档的标准模型
文档--整个网页文档
对象--将网页中的每一个部分都转化为了一个对象
模型--使用模型来表示对象间的关系,这样方便我们获取对像

    (2)tinyxml2的简单示例

待读取的xml文档:

<?xml version="1.0"?>
<root>
<name>MenAngel</name>
</root>

xml文件名为test.xml,代码如下

编译命令 :g++ -g -std=c++11 -I /usr/include main.cpp /usr/include/tinyxml2/tinyxml2.cpp -o test

#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2; int main(int argv,char *argc[])
{
XMLDocument xmlDoc;
xmlDoc.LoadFile("test.xml");
int errorID = xmlDoc.ErrorID();
if(errorID)
{
cout<<"Load xml test.xml fail!"<<endl;
return -;
}
cout<<"Load xml test.xml success!"<<endl;
XMLElement *pRootElement = xmlDoc.RootElement();
const char * name = pRootElement->FirstChildElement("name")->GetText();
cout<<"name = "<<name<<endl;
return ;
}
Load xml test.xml success!
name = MenAngel

    (3)tinyxml2的详细使用范例

通常xml包含如下四个部分:

//节点构成xml的基本单元
文档节点; 对应 XMLDocument
元素节点; 对应 XMLElement
属性节点; 对应 XMLAttribute
文本节点; 对应 XMLText

    1.生成xml文档(增删改查)

#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2;
int createXML(const char * xmlPath)
{
XMLDocument doc;
if(XML_ERROR_FILE_NOT_FOUND != doc.LoadFile(xmlPath))
{
cout<<"file is exits!"<<endl;
return ;
}
//1.添加声明 方法一
//const char* declaration ="<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>";
//doc.Parse(declaration);//会覆盖xml所有内容
//2.添加声明 方法二
//<?xml version="1.0" encoding="UTF-8"?>
XMLDeclaration *declaration = doc.NewDeclaration();
doc.InsertFirstChild(declaration);
//3.新建root根节点
XMLElement *root = doc.NewElement("root");
doc.InsertFirstChild(root);
doc.InsertEndChild(root);
//4.给root添加节点
XMLElement* userNode = doc.NewElement("person");
//4.1设置属性
userNode->SetAttribute("name","MenAngel");
userNode->SetAttribute("passwd",);
//4.2设置开始标签及文本值
userNode->InsertFirstChild(doc.NewText("sunjimeng"));
//4.3设置结束标签
root->InsertEndChild(userNode);
return doc.SaveFile(xmlPath);
}
int addXML(const char *xmlPath)
{
XMLDocument doc;
if( XML_SUCCESS != doc.LoadFile(xmlPath))
{
cout<<"load xml file failed"<<endl;
return -;
}
//再添加一个没有属性的name = MenAngel的 person节点
XMLElement *newPerson = doc.NewElement("person");
newPerson->InsertFirstChild(doc.NewText("MenAngel"));
XMLElement *root = doc.RootElement();
root->InsertEndChild(newPerson);
return doc.SaveFile(xmlPath);
}
int modifyXML(const char *xmlPath)
{
XMLDocument doc;
if( XML_SUCCESS != doc.LoadFile(xmlPath))
{
cout<<"load xml file failed"<<endl;
return -;
}
//1.获得根节点
XMLElement *root = doc.RootElement();
//2.获得根节点root指定名称的第一个子元素
//获得根节点第一个子元素 root->FirstChild();
XMLElement *newPerson = root->FirstChildElement("person");
newPerson->SetAttribute("name","user");
newPerson->SetText("MenAngel");
newPerson->DeleteAttribute("passwd");
//3.查询属性及值
cout<<"<person> name = "<<newPerson->Attribute("name")<<" value = "<<newPerson->GetText()<<endl;
newPerson = newPerson->NextSiblingElement();
root->DeleteChild(newPerson);
return doc.SaveFile(xmlPath);
}
int main(int argv,char *argc[])
{
const char * filename = "create.xml";
int iRet = createXML(filename);
if(iRet != XML_SUCCESS)
cout<<"create xml fail!"<<endl;
iRet = addXML(filename);
if(iRet != XML_SUCCESS)
cout<<"add xml fail!"<<endl;
iRet = modifyXML(filename);
if(iRet != XML_SUCCESS)
cout<<"modify or delte xml fail!"<<endl;
return iRet;
}
//命令行打印
<person> name = user value = MenAngel
//create.xml
<?xml version="1.0" encoding="UTF-8"?>
<root>
<person name="user">MenAngel</person>
</root>

    2.利用for和while循环遍历xml文档中所有文件(两层)

//test.xml
<?xml version="1.0"?>
<root>
<name>MenAngel</name>
<age></age>
<gender>boy</gender>
<hobbys type="array">
<value>语文</value>
<value>数学</value>
<value>英语</value>
</hobbys>
<scores type="array">
<value>
<course>语文</course>
<score></score>
</value>
<value>
<course>数学</course>
<score></score>
</value>
<value>
<course>英语</course>
<score></score>
</value>
</scores>
</root>
//main.cpp
#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2; int main(int argv,char *argc[])
{
XMLDocument doc;
if(XML_SUCCESS != doc.LoadFile("test.xml"))
{
cout<<"load xml file failed!"<<endl;
return -;
}
XMLElement *root = doc.RootElement();
XMLElement *ptrNode = root->FirstChildElement();
do
{ //hobbys标签、scores标签为array类型可以进入
if(ptrNode->Attribute("type") != NULL)
{
cout<<ptrNode->Name()<<" is "<<ptrNode->Attribute("type")<<endl;
//两种标签下均是 value子标签,可以for循环迭代
for(const XMLNode *tempNode = ptrNode->FirstChild();(tempNode != NULL && tempNode->FirstChildElement() == NULL);tempNode = tempNode->NextSibling())
{
cout<<"方法一:"<<tempNode->Value()<<":"<<tempNode->ToElement()->GetText()<<endl;
}
for(XMLElement *ptrElement = ptrNode->FirstChildElement("value");ptrElement; ptrElement=ptrElement->NextSiblingElement("value"))
//hobbys标签下没有子节点,直接是值
if(ptrElement->FirstChildElement() == NULL && ptrElement->GetText() != NULL)
cout<<"方法二:"<<ptrElement->Name()<<":"<<ptrElement->GetText()<<endl;
//scores标签下有两个子节点:course、scores
else
{
cout<<" "<<ptrElement->Value()<<":"<<endl
<<" "<<ptrElement->FirstChildElement("course")->Name()<<":"<<ptrElement->FirstChildElement("course")->GetText()<<endl
<<" "<<ptrElement->FirstChildElement("score")->Name()<<":"<<ptrElement->FirstChildElement("score")->GetText()<<endl;
}
ptrNode = ptrNode->NextSiblingElement();
continue;
}
cout<<ptrNode->Name()<<":"<<ptrNode->GetText()<<endl;
ptrNode = ptrNode->NextSiblingElement();
}while(ptrNode != NULL);
}
//命令行打印
name:MenAngel
age:
gender:boy
hobbys is array
方法一:value:语文
方法一:value:数学
方法一:value:英语
方法二:value:语文
方法二:value:数学
方法二:value:英语
scores is array
value:
course:语文
score:
value:
course:数学
score:
value:
course:英语
score:

    3.利用递归遍历xml文档(小于100层)

#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2; void printBlank(int level = )
{
while(level > )
{
cout<<" ";
level--;
}
}
int printXMLDocument(XMLElement *element,int level = )
{
printBlank(level);
cout<<element->Value()<<":"<<endl;
if(element->FirstChildElement() == NULL)
{
printBlank(level);
cout<<element->Name()<<" is "<<element->GetText()<<endl;
}
for(XMLElement *ptrElement = element->FirstChildElement();ptrElement;ptrElement = ptrElement->NextSiblingElement())
{
printXMLDocument(ptrElement,level + );
}
}
int main(int argv,char *argc[])
{
XMLDocument doc;
if(XML_SUCCESS != doc.LoadFile("test.xml"))
{
cout<<"load xml file failed!"<<endl;
return -;
}
XMLElement *root = doc.RootElement();
printXMLDocument(root);
return ;
}
root:
name:
name is MenAngel
age:
age is
gender:
gender is boy
hobbys:
value:
value is 语文
value:
value is 数学
value:
value is 英语
scores:
value:
course:
course is 语文
score:
score is
value:
course:
course is 数学
score:
score is
value:
course:
course is 英语
score:
score is

    4.打印XMLDocument装载的XML文件

//main.cpp
#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2; int main(int argv,char *argc[])
{
XMLDocument doc;
if(XML_SUCCESS != doc.LoadFile("test.xml"))
{
cout<<"load xml file failed!"<<endl;
return -;
}
XMLPrinter printer;
doc.Accept( &printer );
const char* xmlcstr = printer.CStr();
cout<<xmlcstr<<endl;
return ;
}
//命令行打印
<?xml version="1.0"?>
<root>
<name>MenAngel</name>
<age></age>
<gender>boy</gender>
<hobbys type="array">
<value>语文</value>
<value>数学</value>
<value>英语</value>
</hobbys>
<scores type="array">
<value>
<course>语文</course>
<score></score>
</value>
<value>
<course>数学</course>
<score></score>
</value>
<value>
<course>英语</course>
<score></score>
</value>
</scores>
</root>

    5.查询指定类型的属性

#include <iostream>
#include "tinyxml2/tinyxml2.h"
using namespace std;
using namespace tinyxml2; int main(int argv,char *argc[])
{
XMLDocument doc;
if(XML_SUCCESS != doc.LoadFile("test.xml"))
{
cout<<"load xml file failed!"<<endl;
return -;
}
XMLElement *ptrElement = doc.RootElement()->FirstChildElement("scores")->FirstChildElement("value")->NextSiblingElement();
cout<<"root scores value course = "<<ptrElement->FirstChildElement()->GetText()<<endl;
cout<<"root scores value score = "<<ptrElement->FirstChildElement()->NextSiblingElement()->GetText()<<endl;
int v1;
const char *s1 = new char[];
//查询取值
ptrElement->FirstChildElement("score")->QueryIntText(&v1);
cout<<"root scores value score = "<<v1<<endl;
//查询取属性值
doc.RootElement()->FirstChildElement("scores")->QueryStringAttribute("type",&s1);
cout<<"root scores attribute type = "<<s1<<endl;
return ;
}
root scores  value course = 数学
root scores value score =
root scores value score =
root scores attribute type = array

(3)TinyXml2的详解及使用的更多相关文章

  1. Linq之旅:Linq入门详解(Linq to Objects)

    示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集 ...

  2. 架构设计:远程调用服务架构设计及zookeeper技术详解(下篇)

    一.下篇开头的废话 终于开写下篇了,这也是我写远程调用框架的第三篇文章,前两篇都被博客园作为[编辑推荐]的文章,很兴奋哦,嘿嘿~~~~,本人是个很臭美的人,一定得要截图为证: 今天是2014年的第一天 ...

  3. EntityFramework Core 1.1 Add、Attach、Update、Remove方法如何高效使用详解

    前言 我比较喜欢安静,大概和我喜欢研究和琢磨技术原因相关吧,刚好到了元旦节,这几天可以好好学习下EF Core,同时在项目当中用到EF Core,借此机会给予比较深入的理解,这里我们只讲解和EF 6. ...

  4. Java 字符串格式化详解

    Java 字符串格式化详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 文中如有纰漏,欢迎大家留言指出. 在 Java 的 String 类中,可以使用 format() 方法 ...

  5. Android Notification 详解(一)——基本操作

    Android Notification 详解(一)--基本操作 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Notification 文中如有纰 ...

  6. Android Notification 详解——基本操作

    Android Notification 详解 版权声明:本文为博主原创文章,未经博主允许不得转载. 前几天项目中有用到 Android 通知相关的内容,索性把 Android Notificatio ...

  7. Git初探--笔记整理和Git命令详解

    几个重要的概念 首先先明确几个概念: WorkPlace : 工作区 Index: 暂存区 Repository: 本地仓库/版本库 Remote: 远程仓库 当在Remote(如Github)上面c ...

  8. Drawable实战解析:Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)

    Android XML shape 标签使用详解   一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...

  9. Node.js npm 详解

    一.npm简介 安装npm请阅读我之前的文章Hello Node中npm安装那一部分,不过只介绍了linux平台,如果是其它平台,有前辈写了更加详细的介绍. npm的全称:Node Package M ...

随机推荐

  1. Codeforces Round #454 D. Power Tower (广义欧拉降幂)

    D. Power Tower time limit per test 4.5 seconds memory limit per test 256 megabytes input standard in ...

  2. std::map 的swap错用

    map<int, shared_ptr<int>>map_test; shared_ptr<); map_test[] = tmp_1; shared_ptr<); ...

  3. eclipse+springboot+tomcat自带的部署

    最近在看微服务,然后整理了两个springboot.但执行都是内部main执行,想着后期应该会用到tomcat,大部分都是说的打成war包,然后部署到tomcat上. war包的方式就不说了,网上很多 ...

  4. 3 触发器报警-->远程执行命令

    0.需求 上节课我们讲了,触发器报警,发送邮件,这节课主要讲下远程执行命令 流程图如下 item--> triggers-->action--->Email     |——>远 ...

  5. 下载Americanlife 语音材料

    下载剧本 下载后会在给定目录生成多pdf文件,文件名为每一节的名称 #!/usr/bin/env python3.5 # -*- coding: utf-8 -*- # @Time : 2019/11 ...

  6. 『Codeforces 1186E 』Vus the Cossack and a Field (性质+大力讨论)

    Description 给出一个$n\times m$的$01$矩阵$A$. 记矩阵$X$每一个元素取反以后的矩阵为$X'$,(每一个cell 都01倒置) 定义对$n \times m$的矩阵$A$ ...

  7. Keras学习笔记一:修改数据读入方式为本地图片读入

    第一种方法: Keras官方给的图片去噪示例要自动下载mnist数据集并处理,不能修改和加入自己的数据集. from keras.datasets import mnist (x_train, _), ...

  8. beautifulsoup 安装

    pip install beautifulsoup4

  9. JDK7 JDK8 的安装 且不同版本之间的切换

    myeclipse 论坛下载 https://www.myeclipsecn.com/download/ 用户名:xcj26 邮箱:xcj26@126.com 密码: 26**_X** 版本: Jav ...

  10. Riot.js——一个小而美的JS框架

    Riot.js是什么? Riot 拥有创建现代客户端应用的所有必需的成分: "响应式" 视图层用来创建用户界面 用来在各独立模块之间进行通信的事件库 用来管理URL和浏览器回退按钮 ...