TinyXML教程

TinyXML教程

这是什么?

这个教程对如何有效地使用TinyXML有几个技巧与建议。

我也会尽量包含一些C++的小技巧,如如何将string类型转换为整形和反过来转换。这并不是只针对TinyXML自身,只是因为它或许对你的项目有用,所以我才放在本文的一些地方。

如果你并不了解基本的C++概念,这篇教程对你并没有用。同样地,如果你不知道DOM是什么,先到别的地方了解一下。

在开始之前

一些将要被用到的XML数据组或文件。

example1.xml:

<?xmlversion="1.0" ?>

<Hello>World</Hello>

example2.xml:

<?xml version="1.0" ?>
<poetry>
        <verse>
                 Alas
                   Great World
                          Alas (again)
        </verse>
</poetry>

example3.xml:

<?xml version="1.0" ?>
<shapes>
        <circle name="int-based" x="20" y="30" r="50" />
        <point name="float-based" x="3.5" y="52.1" />
</shapes>

example4.xml

<?xml version="1.0" ?>
<MyApp>
    <!-- Settings for MyApp -->
    <Messages>
        <Welcome>Welcome to MyApp</Welcome>
        <Farewell>Thank you for using MyApp</Farewell>
    </Messages>
    <Windows>
        <Window name="MainFrame" x="5" y="15" w="400" h="250" />
    </Windows>
    <Connection ip="192.168.0.1" timeout="123.456000" />
</MyApp>

开始

从文件中载入XML

从文件导入到TinyXML DOM的最简单方法就是:

TiXmlDocument doc( "demo.xml" );
doc.LoadFile();

下面展示了一个更实用的方法。它会载入文件并且将文件内容显示到STDOUT。

// load the named file and dump its structure to STDOUT
void dump_to_stdout(const char* pFilename)
{
        TiXmlDocument doc(pFilename);
        bool loadOkay = doc.LoadFile();
        if (loadOkay)
        {
                 printf("\n%s:\n", pFilename);
                 dump_to_stdout( &doc ); // defined later in the tutorial
        }
        else
        {
                 printf("Failed to load file \"%s\"\n", pFilename);
        }
}

最简单的证实这个函数的方法是使用像下面一样的main函数:

int main(void)
{
        dump_to_stdout("example1.xml");
        return 0;
}

回顾Example 1中的XML:

<?xml version="1.0" ?>
<Hello>World</Hello>

使用这个XML运行上面的程序将在控制台/DOS窗口中输出如下的结果:

DOCUMENT
+ DECLARATION
+ ELEMENT Hello
  + TEXT[World]

“dump_to_stdout”函数会在本教程的下面重新定义,如果你想理解递归遍历一个DOM,这将非常有用。

程序化创建文档

这是如何程序化构建Example 1:

void build_simple_doc( )
{
        // Make xml: <?xml ..><Hello>World</Hello>
        TiXmlDocument doc;
        TiXmlDeclaration * decl = new TiXmlDeclaration( "1.0", "", "" );
        TiXmlElement * element = new TiXmlElement( "Hello" );
        TiXmlText * text = new TiXmlText( "World" );
        element->LinkEndChild( text );
        doc.LinkEndChild( decl );
        doc.LinkEndChild( element );
        doc.SaveFile( "madeByHand.xml" );
}

这个文件可以通过如下方法载入并且显示在控制台:

dump_to_stdout("madeByHand.xml"); // 这个函数将会在本教程后续定义

你会发现它与Example 1相同:

madeByHand.xml:
Document
+ Declaration
+ Element [Hello]
  + Text: [World]

这个代码生成完全相同的XML DOM,但是它却展示了创建与连接节点的不同顺序:

void write_simple_doc2( )
{
        // same as write_simple_doc1 but add each node
        // as early as possible into the tree.
 
        TiXmlDocument doc;
        TiXmlDeclaration * decl = new TiXmlDeclaration( "1.0", "", "" );
        doc.LinkEndChild( decl );
        
        TiXmlElement * element = new TiXmlElement( "Hello" );
        doc.LinkEndChild( element );
        
        TiXmlText * text = new TiXmlText( "World" );
        element->LinkEndChild( text );
        
        doc.SaveFile( "madeByHand2.xml" );
}

这些都产生相同的XML:

<?xml version="1.0" ?>
<Hello>World</Hello>

或者相同的结构:

DOCUMENT
+ DECLARATION
+ ELEMENT Hello
  + TEXT[World]

属性(Attributes)

给一个存在的节点设置属性是简单的:

window = new TiXmlElement( "Demo" );  
window->SetAttribute("name", "Circle");
window->SetAttribute("x", 5);
window->SetAttribute("y", 15);
window->SetDoubleAttribute("radius", 3.14159);

只要你想,你就可以让它通过TiXmlAttribute类实现。

下面的代码展示了一个(并不是唯一)获取节点所有属性的方法,打印名称和字符值,并且如果这些值能够转换为整型或double类型,打印这些值:

// print all attributes of pElement.
// returns the number of attributes printed
int dump_attribs_to_stdout(TiXmlElement* pElement, unsigned int indent)
{
   if ( !pElement ) return 0;
 
   TiXmlAttribute* pAttrib=pElement->FirstAttribute();
   int i=0;
   int ival;
   double dval;
   const char* pIndent=getIndent(indent);
   printf("\n");
   while (pAttrib)
   {
printf( "%s%s: value=[%s]", pIndent, pAttrib->Name(), pAttrib->Value());
 
       if (pAttrib->QueryIntValue(&ival)==TIXML_SUCCESS)  printf( " int=%d", ival);
       if (pAttrib->QueryDoubleValue(&dval)==TIXML_SUCCESS) printf( " d=%1.1f", dval);
       printf( "\n" );
       i++;
       pAttrib=pAttrib->Next();
   }
   return i;
}

将文档写入文件

将一个预先构建的DOM写入文件是个简单的事:

doc.SaveFile( saveFilename );

回忆,例如example 4:

<?xml version="1.0" ?>
<MyApp>
    <!-- Settings for MyApp -->
    <Messages>
        <Welcome>Welcome to MyApp</Welcome>
        <Farewell>Thank you for using MyApp</Farewell>
    </Messages>
    <Windows>
        <Window name="MainFrame" x="5" y="15" w="400" h="250" />
    </Windows>
    <Connection ip="192.168.0.1" timeout="123.456000" />
</MyApp>

如下的函数创建了这个DOM并且写入”appsettings.xml“文件:

void write_app_settings_doc( )  
        TiXmlDocument doc;  
        TiXmlElement* msg;
         TiXmlDeclaration* decl = new TiXmlDeclaration( "1.0", "", "" );  
        doc.LinkEndChild( decl );  
 
        TiXmlElement * root = new TiXmlElement( "MyApp" );  
        doc.LinkEndChild( root );  
 
        TiXmlComment * comment = new TiXmlComment();
        comment->SetValue(" Settings for MyApp " );  
        root->LinkEndChild( comment );  
 
        TiXmlElement * msgs = new TiXmlElement( "Messages" );  
        root->LinkEndChild( msgs );  
 
        msg = new TiXmlElement( "Welcome" );  
        msg->LinkEndChild( new TiXmlText( "Welcome to MyApp" ));  
        msgs->LinkEndChild( msg );  
 
        msg = new TiXmlElement( "Farewell" );  
        msg->LinkEndChild( new TiXmlText( "Thank you for using MyApp" ));  
        msgs->LinkEndChild( msg );  
 
        TiXmlElement * windows = new TiXmlElement( "Windows" );  
        root->LinkEndChild( windows );  
 
        TiXmlElement * window;
        window = new TiXmlElement( "Window" );  
        windows->LinkEndChild( window );  
        window->SetAttribute("name", "MainFrame");
        window->SetAttribute("x", 5);
        window->SetAttribute("y", 15);
        window->SetAttribute("w", 400);
        window->SetAttribute("h", 250);
 
        TiXmlElement * cxn = new TiXmlElement( "Connection" );  
        root->LinkEndChild( cxn );  
        cxn->SetAttribute("ip", "192.168.0.1");
        cxn->SetDoubleAttribute("timeout", 123.456); // floating point attrib
        
        dump_to_stdout( &doc );
        doc.SaveFile( "appsettings.xml" );  
} 

函数dump_to_stdout将显示这个结构:

Document
+ Declaration
+ Element [MyApp]
 (No attributes)
  + Comment: [ Settings for MyApp ]
  + Element [Messages]
 (No attributes)
    + Element [Welcome]
 (No attributes)
      + Text: [Welcome to MyApp]
    + Element [Farewell]
 (No attributes)
      + Text: [Thank you for using MyApp]
  + Element [Windows]
 (No attributes)
    + Element [Window]
      + name: value=[MainFrame]
      + x: value=[5] int=5 d=5.0
      + y: value=[15] int=15 d=15.0
      + w: value=[400] int=400 d=400.0
      + h: value=[250] int=250 d=250.0
      5 attributes
  + Element [Connection]
    + ip: value=[192.168.0.1] int=192 d=192.2
    + timeout: value=[123.456000] int=123 d=123.5
    2 attributes

我对TinyXML默认地按照别的APIs所谓的”整洁“方式写XML文档感到奇怪。它改变了包含别的节点的元素文本的空白字符,因此它写成了包含层次嵌套的树状。

我并没有看到有方法去关闭写文件时的缩进,使界限相对简单。

[Lee:这在STL模式时很简单,仅仅只需使用cout<< myDoc。非STL模式总是以“整洁”模式显示。添加一个开关是一个非常好的功能并且正在考虑中。]

XML到C++对象/C++对象到XML

介绍

这个例子结舌你已经学过了从XML文件中导入程序设置与将程序配置写入XML文件,如类似example4.xml。

有许多实现的方法。例如,研究TinyBind项目(http://sourceforge.net/projects/tinybind)。

这个章节介绍一个使用XML读入与写出基本对象结构的悠久方法。

设置你的类对象

以几个基本类开始,就像:

#include <string>
#include <map>
using namespace std;
 
typedef std::map<std::string,std::string> MessageMap;
 
// a basic window abstraction - demo purposes only
class WindowSettings
{
public:
        int x,y,w,h;
        string name;                               
 
        WindowSettings()
                 : x(0), y(0), w(100), h(100), name("Untitled")
        {
        }
 
        WindowSettings(int x, int y, int w, int h, const string& name)
        {
                 this->x=x;
                 this->y=y;
                 this->w=w;
                 this->h=h;
                 this->name=name;
        }
};
 
class ConnectionSettings
{
public:
        string ip;
        double timeout;
};
 
class AppSettings
{
public:
        string m_name;
        MessageMap m_messages;
        list<WindowSettings> m_windows;
        ConnectionSettings m_connection;
 
        AppSettings() {}
 
        void save(const char* pFilename);
        void load(const char* pFilename);
        
        // just to show how to do it
        void setDemoValues()
        {
                 m_name="MyApp";
                 m_messages.clear();
                 m_messages["Welcome"]="Welcome to "+m_name;
                 m_messages["Farewell"]="Thank you for using "+m_name;
                 m_windows.clear();
                 m_windows.push_back(WindowSettings(15,15,400,250,"Main"));
                 m_connection.ip="Unknown";
                 m_connection.timeout=123.456;
        }
};

这是一个基本的main函数,它展示了如何创建一个节本的对象树,并随后保存和载入:

int main(void)
{
        AppSettings settings;
        
        settings.save("appsettings2.xml");
        settings.load("appsettings2.xml");
        return 0;
}

下面的main函数展示创建、修改、保存和载入一个预设的结构:

int main(void)
{
        // block: customise and save settings
        {
                 AppSettings settings;
                 settings.m_name="HitchHikerApp";
                 settings.m_messages["Welcome"]="Don't Panic";
                 settings.m_messages["Farewell"]="Thanks for all the fish";
                 settings.m_windows.push_back(WindowSettings(15,25,300,250,"BookFrame"));
                 settings.m_connection.ip="192.168.0.77";
                 settings.m_connection.timeout=42.0;
 
                 settings.save("appsettings2.xml");
        }
        
        // block: load settings
        {
                 AppSettings settings;
                 settings.load("appsettings2.xml");
                 printf("%s: %s\n", settings.m_name.c_str(), 
                          settings.m_messages["Welcome"].c_str());
                 WindowSettings & w=settings.m_windows.front();
                 printf("%s: Show window '%s' at %d,%d (%d x %d)\n", 
                          settings.m_name.c_str(), w.name.c_str(), w.x, w.y, w.w, w.h);
                 printf("%s: %s\n", settings.m_name.c_str(), settings.m_messages["Farewell"].c_str());
        }
        return 0;
}

当save()与load()结束时(看下面),运行main()在控制台上显示:

HitchHikerApp: Don't Panic
HitchHikerApp: Show window 'BookFrame' at 15,25 (300 x 100)
HitchHikerApp: Thanks for all the fish

C++ XML编码

实现保存到一个文件有很多种方法。这就是其中之一:

void AppSettings::save(const char* pFilename)
{
        TiXmlDocument doc;  
        TiXmlElement* msg;
        TiXmlComment * comment;
        string s;
         TiXmlDeclaration* decl = new TiXmlDeclaration( "1.0", "", "" );  
        doc.LinkEndChild( decl ); 
 
        TiXmlElement * root = new TiXmlElement(m_name.c_str());  
        doc.LinkEndChild( root );  
 
        comment = new TiXmlComment();
        s=" Settings for "+m_name+" ";
        comment->SetValue(s.c_str());  
        root->LinkEndChild( comment );  
 
        // block: messages
        {
                 MessageMap::iterator iter;
 
                 TiXmlElement * msgs = new TiXmlElement( "Messages" );  
                 root->LinkEndChild( msgs );  
 
                 for (iter=m_messages.begin(); iter != m_messages.end(); iter++)
                 {
                          const string & key=(*iter).first;
                          const string & value=(*iter).second;
                          msg = new TiXmlElement(key.c_str());  
                          msg->LinkEndChild( new TiXmlText(value.c_str()));  
                          msgs->LinkEndChild( msg );  
                 }
        }
 
        // block: windows
        {
                 TiXmlElement * windowsNode = new TiXmlElement( "Windows" );  
                 root->LinkEndChild( windowsNode );  
 
                 list<WindowSettings>::iterator iter;
 
                 for (iter=m_windows.begin(); iter != m_windows.end(); iter++)
                 {
                          const WindowSettings& w=*iter;
 
                          TiXmlElement * window;
                          window = new TiXmlElement( "Window" );  
                          windowsNode->LinkEndChild( window );  
                          window->SetAttribute("name", w.name.c_str());
                          window->SetAttribute("x", w.x);
                          window->SetAttribute("y", w.y);
                          window->SetAttribute("w", w.w);
                          window->SetAttribute("h", w.h);
                 }
        }
 
        // block: connection
        {
                 TiXmlElement * cxn = new TiXmlElement( "Connection" );  
                 root->LinkEndChild( cxn );  
                 cxn->SetAttribute("ip", m_connection.ip.c_str());
                 cxn->SetDoubleAttribute("timeout", m_connection.timeout); 
        }
 
        doc.SaveFile(pFilename);  
}

修改main函数并且运行它会产生如下的文件:

<?xml version="1.0" ?>
<HitchHikerApp>
    <!-- Settings for HitchHikerApp -->
    <Messages>
        <Farewell>Thanks for all the fish</Farewell>
        <Welcome>Don&apos;t Panic</Welcome>
    </Messages>
    <Windows>
        <Window name="BookFrame" x="15" y="25" w="300" h="250" />
    </Windows>
    <Connection ip="192.168.0.77" timeout="42.000000" />
</HitchHikerApp>

XML解码

与对象编码类似,有很多方法将XML解码成C++对象结构。下面的方法使用了TiXmlHandles:

void AppSettings::load(const char* pFilename)
{
        TiXmlDocument doc(pFilename);
        if (!doc.LoadFile()) return;
 
        TiXmlHandle hDoc(&doc);
        TiXmlElement* pElem;
        TiXmlHandle hRoot(0);
 
        // block: name
        {
                 pElem=hDoc.FirstChildElement().Element();
                 // should always have a valid root but handle gracefully if it does
                 if (!pElem) return;
                 m_name=pElem->Value();
 
                 // save this for later
                 hRoot=TiXmlHandle(pElem);
        }
 
        // block: string table
        {
                 m_messages.clear(); // trash existing table
 
                 pElem=hRoot.FirstChild( "Messages" ).FirstChild().Element();
                 for( pElem; pElem; pElem=pElem->NextSiblingElement())
                 {
                          const char *pKey=pElem->Value();
                          const char *pText=pElem->GetText();
                          if (pKey && pText) 
                          {
                                   m_messages[pKey]=pText;
                          }
                 }
        }
 
        // block: windows
        {
                 m_windows.clear(); // trash existing list
 
                 TiXmlElement* pWindowNode=hRoot.FirstChild( "Windows" ).FirstChild().Element();
                 for( pWindowNode; pWindowNode; pWindowNode=pWindowNode->NextSiblingElement())
                 {
                          WindowSettings w;
                          const char *pName=pWindowNode->Attribute("name");
                          if (pName) w.name=pName;
                          
                          pWindowNode->QueryIntAttribute("x", &w.x); // If this fails, original value is left as-is
                          pWindowNode->QueryIntAttribute("y", &w.y);
                          pWindowNode->QueryIntAttribute("w", &w.w);
                          pWindowNode->QueryIntAttribute("hh", &w.h);
 
                          m_windows.push_back(w);
                 }
        }
 
        // block: connection
        {
                 pElem=hRoot.FirstChild("Connection").Element();
                 if (pElem)
                 {
                          m_connection.ip=pElem->Attribute("ip");
                          pElem->QueryDoubleAttribute("timeout",&m_connection.timeout);
                 }
        }
}

完整的dump_to_stdout

下面是一个可以复制与粘贴的程序例子,它可以载入XML属性和通过前面提到的递归遍历方式将这种结构展示到STDOUT。

// tutorial demo program
#include "stdafx.h"
#include "tinyxml.h"
 
// ----------------------------------------------------------------------
// STDOUT dump and indenting utility functions
// ----------------------------------------------------------------------
const unsigned int NUM_INDENTS_PER_SPACE=2;
 
const char * getIndent( unsigned int numIndents )
{
        static const char * pINDENT="                                      + ";
        static const unsigned int LENGTH=strlen( pINDENT );
        unsigned int n=numIndents*NUM_INDENTS_PER_SPACE;
        if ( n > LENGTH ) n = LENGTH;
 
        return &pINDENT[ LENGTH-n ];
}
 
// same as getIndent but no "+" at the end
const char * getIndentAlt( unsigned int numIndents )
{
        static const char * pINDENT="                                        ";
        static const unsigned int LENGTH=strlen( pINDENT );
        unsigned int n=numIndents*NUM_INDENTS_PER_SPACE;
        if ( n > LENGTH ) n = LENGTH;
 
        return &pINDENT[ LENGTH-n ];
}
 
int dump_attribs_to_stdout(TiXmlElement* pElement, unsigned int indent)
{
        if ( !pElement ) return 0;
 
        TiXmlAttribute* pAttrib=pElement->FirstAttribute();
        int i=0;
        int ival;
        double dval;
        const char* pIndent=getIndent(indent);
        printf("\n");
        while (pAttrib)
        {
                 printf( "%s%s: value=[%s]", pIndent, pAttrib->Name(), pAttrib->Value());
 
                 if (pAttrib->QueryIntValue(&ival)==TIXML_SUCCESS)    printf( " int=%d", ival);
                 if (pAttrib->QueryDoubleValue(&dval)==TIXML_SUCCESS) printf( " d=%1.1f", dval);
                 printf( "\n" );
                 i++;
                 pAttrib=pAttrib->Next();
        }
        return i;        
}
 
void dump_to_stdout( TiXmlNode* pParent, unsigned int indent = 0 )
{
        if ( !pParent ) return;
 
        TiXmlNode* pChild;
        TiXmlText* pText;
        int t = pParent->Type();
        printf( "%s", getIndent(indent));
        int num;
 
        switch ( t )
        {
        case TiXmlNode::TINYXML_DOCUMENT:
                 printf( "Document" );
                 break;
 
        case TiXmlNode::TINYXML_ELEMENT:
                 printf( "Element [%s]", pParent->Value() );
                 num=dump_attribs_to_stdout(pParent->ToElement(), indent+1);
                 switch(num)
                 {
                          case 0:  printf( " (No attributes)"); break;
                          case 1:  printf( "%s1 attribute", getIndentAlt(indent)); break;
                          default: printf( "%s%d attributes", getIndentAlt(indent), num); break;
                 }
                 break;
 
        case TiXmlNode::TINYXML_COMMENT:
                 printf( "Comment: [%s]", pParent->Value());
                 break;
 
        case TiXmlNode::TINYXML_UNKNOWN:
                 printf( "Unknown" );
                 break;
 
        case TiXmlNode::TINYXML_TEXT:
                 pText = pParent->ToText();
                 printf( "Text: [%s]", pText->Value() );
                 break;
 
        case TiXmlNode::TINYXML_DECLARATION:
                 printf( "Declaration" );
                 break;
        default:
                 break;
        }
        printf( "\n" );
        for ( pChild = pParent->FirstChild(); pChild != 0; pChild = pChild->NextSibling()) 
        {
                 dump_to_stdout( pChild, indent+1 );
        }
}
 
// load the named file and dump its structure to STDOUT
void dump_to_stdout(const char* pFilename)
{
        TiXmlDocument doc(pFilename);
        bool loadOkay = doc.LoadFile();
        if (loadOkay)
        {
                 printf("\n%s:\n", pFilename);
                 dump_to_stdout( &doc ); // defined later in the tutorial
        }
        else
        {
                 printf("Failed to load file \"%s\"\n", pFilename);
        }
}
 
// ----------------------------------------------------------------------
// main() for printing files named on the command line
// ----------------------------------------------------------------------
int main(int argc, char* argv[])
{
        for (int i=1; i<argc; i++)
        {
                 dump_to_stdout(argv[i]);
        }
        return 0;
}

在命令行或者DOS窗口中运行,如下:

C:\dev\tinyxml> Debug\tinyxml_1.exe example1.xml
 
example1.xml:
Document
+ Declaration
+ Element [Hello]
 (No attributes)
  + Text: [World]

注:本文翻译自TinyXML的源码文档,如果不恰当之处,请不吝赐教,谢谢!
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章