tinyxml_2_6_1.tar.gz安装与移植

一、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的类图,反应各个类之间的静态关系。 

tinyxml_2_6_1.tar.gz安装与移植_第1张图片

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();  
//_____________________________________________________________  

你可能感兴趣的:(XML相关开发)