一、TinyXml的特点
TinyXml是一个简单、小巧的C++XML解释器。TinyXml具有以下特点:
① TinyXml最大的特点就是体积小,速度快,而且只有2个头文件和3个源文件,总共144 KB,十分适合嵌入式系统应用。
② 功能完善,简单易用,为用户提供了丰富的API。TinyXml是一种基于DOM模型的解释器。基于DOM模型是指在分析时,一次性对整个XML文档进行分析,并在内存中形成对应的树结构;同时,向用户提供一系列的接口来访问和编辑该树结构。这种方式占用内存大,速度往往较慢,但可以给用户提供一个面向对象的访问接口,对用户更为友好。在本项目中需要存放的参数不是很多,因而不会占用系统很多的内存。
③ 源码开放,无须付费。开放的源码不仅可以降低产品的生产成本,更为重要的是为产品的维护完善和稳定运行提供了最为彻底的保障。
④ 易于移植。TinyXml与操作系统无关,可以方便地移植到各种系统平台。源代码中已经提供了一份Make-file文件,稍加改造就可以移植到我们的嵌入式Linux下
二、 TinyXml在嵌入式Linux下的移植
操作系统:centos6.5
位数:64位
tinyxml_2_6_1.tar.gz下载地址:http://download.csdn.net/detail/lqpbeyond/2280080#comment
(1) 建立arm-linux交叉编译环境
http://blog.csdn.net/qq_22790049/article/details/52806203
(2)解压源码
#tar -xzvf tinyxml_2_6_1.tar.gz
(3)进入源码目录修改Makefile
#cd tinyxml
#vim Makefile
修改如下:
① 将"CXX:=g++"
改为"CXX:=arm-linux-g++"
,即将编译器由g++改为交叉编译器。
② 将"OUTPUT:=xmltest"
改为"OUTPUT:=libtinyxml.so"
,即将原来生成演示程序改为生成动态链接库libtinyxml.so。
③ 将xmltest.cpp从"SRCS:=tinyxml.cpp tinyxml-parser.cpp xmltest.cpp tinyxmlerror.cpp tinystr.cpp"
中删除,注释掉"xmltest.o:tinyxml.h tinystr.h"
。因为不需要将演示程序添加到动态库中。
④ 在"${LD}-o$@${LDFLAGS)${OBJS}${LIBS}${EXTRA_LIBS}"
语句的”${LD)”后添加”-shared”,用于生成动态库。
(4) 生成动态链接库libtinyxml.so
修改完成后,执行make命令就可以在当前目录生成需要的动态链接库文件libtinyxml.so,也可以根据需要把TinyXml编译成静态库文件。
#make
出现如下错误:
/usr/bin/ld: tinyxml.o: Relocations in generic ELF (EM: 40)
tinyxml.o: could not read symbols: File in wrong format
通过查找资料发现,原因是编译工具不一致,分析make后的编译信息可知g++ -shared -o tinyxml.so tinyxml.o tinyxmlparser.o tinyxmlerror.o tinystr.o
,与我们指定的交叉编译工具链不一致,要修改Makefile,修改如下:
将"LD:=g++"
改为"LD:=arm-linux-g++"
重新编译即可在当前文件夹下生成tinyxml.so文件。
三、tinyxml的使用
在tinyxml中主要文件这里写代码片
分别是tinystr.h、tinystr.cpp、tinyxml.h、tinyxml.cpp、tinyxmlerror.cpp、tinyxmlparser.cpp。在需要操作xml文件的地方,使用如下代码,就可以引入tinyXML类库。
#include
或者
#include "tinyxml.h"
下面我用个简单的例子说明如何使用tinyXML操作xml文件。在讲例子之前我先说说tinyXML中主要类和xml文档之间的对应关系。下面是tinyXML中主要class的类图,反应各个类之间的静态关系。
TiXmlBase是所有类的基类,TiXmlNode、TiXmlAttribute两个类都继承来自TiXmlBase类,其中TiXmlNode类指的是所有被<...>...<.../>包括的内容,而xml中的节点又具体分为以下几方面内容,分别是声明、注释、节点以及节点间的文本,因此在TiXmlNode的基础上又衍生出这几个类TiXmlComment、TiXmlDeclaration、TiXmlDocument、TiXmlElement、TiXmlText、TiXmlUnknown,分别用来指明具体是xml中的哪一部分。TiXmlAttribute类不同于TiXmlNode,它指的是在尖括号里面的内容,像<... ***=...>,其中***就是一个属性。这块我具体用一个xml文档说明一下,内容如下:
Xml代码
<phonebook>
<item>
<name>miaomaioname>
<addr>Shaanxi Xi'anaddr>
<tel>13759911917tel>
<email>[email protected]email>
item>
<item>
<name>gougouname>
<addr>Liaoning Shenyangaddr>
<tel>15840330481tel>
<email>[email protected]email>
item>
phonebook>
像TiXmlDeclaration指的就是,
像TiXmlComment指的就是、
,
像TiXmlDocument指的就是整个xml文档,
像TiXmlElement指的就是
等等这些节点,
像TiXmlText指的就是‘gougou’、‘15840330481’这些夹在与、与、与之间的文本文字,
像TiXmlAttribute指的就是节点中version、encoding,
除此之外就是TiXmlUnknown。
下面是我自己写的一段读xml文件的c++代码,以及再往xml写入一个item的源代码,其中phonebookdata.xml中的内容就是上面xml,仅供参考。
C++代码
//______________________________________________________
// Read information from xml file.
// define xml file path, as follow , we use relative path,
// but you can use absolute path also.
const char* filepath = "phonebookdata.xml";
TiXmlDocument doc(filepath);
bool loadOkay = doc.LoadFile();
// faile to load 'phonebookdata.xml'.
if (!loadOkay) {
printf( "Could not load test file %s. Error='%s'. Exiting.\n", filepath,doc.ErrorDesc() );
exit( 1 );
}
// get dom root of 'phonebookdata.xml', here root should be 'phonebook'.
TiXmlElement* root = doc.RootElement();
printf("_______________________________________\n\n");
printf(" contacted person information \n\n");
// trace every items below root.
for( TiXmlNode* item = root->FirstChild( "item" );
item;
item = item->NextSibling( "item" ) ) {
printf("_______________________________________\n");
// read name.
TiXmlNode* child = item->FirstChild();
const char* name = child->ToElement()->GetText();
if (name) {
printf("name:%s\n",name);
} else {
printf("name:\n");
}
// read address.
child = item->IterateChildren(child);
const char* addr = child->ToElement()->GetText();
if (addr) {
printf("addr:%s\n",addr);
} else {
printf("addr:\n");
}
// read telephone no.
child = item->IterateChildren(child);
const char* tel = child->ToElement()->GetText();
if (tel) {
printf("tel:%s\n",tel);
} else {
printf("tel:\n");
}
// read e-mail.
child = item->IterateChildren(child);
const char* email = child->ToElement()->GetText();
if(email) {
printf("email:%s\n",email);
} else {
printf("email:\n");
}
printf("\n");
}
//_____________________________________________________________
//_____________________________________________________________
// Add information to xml file and save it.
TiXmlElement* writeRoot = doc.RootElement();
TiXmlNode* newNode = new TiXmlElement("item");
const TiXmlNode* name4NewNode = new TiXmlElement("name");
newNode->InsertEndChild(*name4NewNode)->InsertEndChild(TiXmlText("pipi"));
const TiXmlNode* addr4NewNode = new TiXmlElement("addr");
newNode->InsertEndChild(*addr4NewNode)->InsertEndChild(TiXmlText("Shaanxi Xianyang"));
const TiXmlNode* tel4NewNode = new TiXmlElement("tel");
newNode->InsertEndChild(*tel4NewNode)->InsertEndChild(TiXmlText("02937310627"));
const TiXmlNode* email4NewNode = new TiXmlElement("email");
newNode->InsertEndChild(*email4NewNode)->InsertEndChild(TiXmlText("[email protected]"));
writeRoot->InsertEndChild(*newNode);
doc.SaveFile();
//_____________________________________________________________