1.TinyXML2概述
TinyXML2是simple、small、efficient开源的C++ XML文件解析库,可以很方便的应用到现有的项目之中。非常适合存储简单数据,配置文件,对象序列化等数据量不是很大的操作。
TinyXML2详细介绍与源码获取方法详见:TinyXML2官网。
2. TinyXML1与TinyXML2对比
TinyXML1与TinyXML2这两个著名的开源XML文件解析库均出自Lee Thomason之手,向这位满怀开源精神的大家致敬。
TinyXML2适用于大部分的C/C++项目开发,经得住考验,是最好的选择。较TinyXML1而言,TinyXML2化繁为简,使用时只需要包含两个文件,而TinyXML1需要6个文件,一般生成静态链接库供项目的使用。TinyXML1详细介绍与源码见:TinyXML1官网。TinyXML1用法用例可以参考博文:TinyXML快速入门。
TinyXML2使用了与TinyXML1相似都可API,并且拥有丰富的测试案例。但TinyXML2解析器相对TinyXML1在代码上是完全重写,使其更适合于游戏开发中使用。它使用更少的内存,更快,并使用更少的内存分配。
TinyXML2无需STL,也放弃了对STL支持。所有字符串查询均使用C风格字符串“const char *”来表示,省去string类型对象的构造,并使代码更简单。
二者共同点: (1)都使用了简单易用的API。 (2)都是基于DOM(Document Object Model,文档对象模型)的解析器。 (3)都支持UTF-8编码。
TinyXML2的优点: (1)对大部分大部分的C/C++项目具有普适性。 (2)使用较少的内存(约TinyXML1的40%),速度变得更快。 (3)没有C++的STL的要求。 (4)更接近现代C++的特性,如使用了适当的命名空间。 (5)适当有效的处理了的空白字符(空格,TAB和回车)。
TinyXML1的优点: (1)可以报告分析错误的位置。 (2)提供一些C++ STL公约支持:流和字符串。 (3)拥有非常成熟和良好的调试代码库。
3. TinyXML2的用法用例
TinyXML2的网上教程并不多见,醍醐灌顶,受益匪浅的教程更是凤毛麟角。有的也是蜻蜓点水、参差不齐的泛泛而谈。最终,所能参考的资料也就是官网的文档和示例代码,但却有点晦涩难懂。因此,本文就为了解决这个尴尬的局面,结合官网的资料和网上资源,尽量详细的列出TinyXML2的常见用法用例,不足之处,请留言补充,后续增加修改。
xml文件本质就是小型的数据库,换个角度来说就是,对数据库有什么操作,那么对xml文件就应能实现什么操作。一般而言,对数据库的操作包括以下几种:新建数据库和对数据库增删查改。那么对应xml文件就是新建xml文件、增加xml文件的节点,删除xml文件的指定节点,查询xml文件指定节点的值,修改xml文件中节点的值。
使用方法:将tinyxml2.cpp和tinyxml2.h拷贝至项目目录,使用时包含#include “tinyxml2.h”和using namespace tinyxml2。
使用场景:存储用户信息。
用户数据表设计如下:
1 2 3 4 5 6
| 变量名 描述 类型 长度(字节) 不为空 主键 UserName 用户名 Vchar 3-20 Y Y Password 密码 Char 32 Y N Gender 性别 Int 1 N N Mobile 电话 Char 11 N N Email 电子邮箱 Varchar 1-50 N N
|
对应XML文件实现如下:
1 2 3 4 5 6 7 8 9
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> <DBUSER> <User Name=”lvlv” Password =”123456”> <Gender></Gender> <Mobile ></ Mobile> <Email ></ Email > </User>
<DBUSER>
|
从中可以看出,XML由三大部分组成,分别是声明、根节点和其它节点。其中xml文件的声明包括三方面的内容:Version、Standalone和Encoding。下面将详细列出常见tinyxml2的用法。
注意:以下示例代码针对本人下载使用的TinyXML2,官网的TinyXML2在不断的完善和更新当中,最新的TinyXML2和本人的示例代码可能会有出入。本人使用的TinyXML2是2015.9.23从官网下载的,已上传至CSDN下载,见:TinyXML2。
3.1创建XML文件
示例代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
|
int createXML(const char* xmlPath) { const char* declaration ="<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>"; XMLDocument doc; doc.Parse(declaration);
XMLElement* root=doc.NewElement("DBUSER"); doc.InsertEndChild(root);
return doc.SaveFile(xmlPath); }
|
创建结果:
1 2
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> <DBUSER/>
|
3.2增加xml文件的节点
示例代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
| class User { public: User(){ gender=0; };
User(const string& userName, const string& password, int gender, const string& mobile, const string& email){ this->userName=userName; this->password=password; this->gender=gender; this->mobile=mobile; this->email=email; };
string userName; string password; int gender; string mobile; string email; };
int insertXMLNode(const char* xmlPath,const User& user) { XMLDocument doc; int res=doc.LoadFile(xmlPath); if(res!=0) { cout<<"load xml file failed"<<endl; return res; } XMLElement* root=doc.RootElement();
XMLElement* userNode = doc.NewElement("User"); userNode->SetAttribute("Name",user.userName.c_str()); userNode->SetAttribute("Password ",user.password.c_str()); root->InsertEndChild(userNode);
XMLElement* gender = doc.NewElement("Gender"); XMLText* genderText=doc.NewText(itoa(user.gender)); gender->InsertEndChild(genderText); userNode->InsertEndChild(gender);
XMLElement* mobile = doc.NewElement("Mobile"); mobile->InsertEndChild(doc.NewText(user.mobile.c_str())); userNode->InsertEndChild(mobile);
XMLElement* email = doc.NewElement("Email"); email->InsertEndChild(doc.NewText(user.email.c_str())); userNode->InsertEndChild(email);
return doc.SaveFile(xmlPath); }
|
创建结果:
1 2 3 4 5 6 7 8
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> <DBUSER> <User Name="lvlv" Password ="12346"> <Gender>1</Gender> <Mobile>15813354926</Mobile> <Email>1589276509@qq.com</Email> </User> </DBUSER>
|
3.3查询xml文件的指定节点
Xml文件中,一个用户节点存储一个用户的信息。因此,对用户信息的增删查改,即无论查询节点、删除节点、修改节点和增加节点,都需要获取需要操作的节点。那么先实现一个根据用户名获取节点指针的函数:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
|
XMLElement* queryUserNodeByName(XMLElement* root,const string& userName) {
XMLElement* userNode=root->FirstChildElement("User"); while(userNode!=NULL) { if(userNode->Attribute("Name")==userName) break; userNode=userNode->NextSiblingElement(); } return userNode; }
|
在以上函数的基础上,获取用户信息的函数:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| User* queryUserByName(const char* xmlPath,const string& userName) { XMLDocument doc; if(doc.LoadFile(xmlPath)!=0) { cout<<"load xml file failed"<<endl; return NULL; } XMLElement* root=doc.RootElement(); XMLElement* userNode=queryUserNodeByName(root,userName);
if(userNode!=NULL) { User* user=new User(); user->userName=userName; user->password=userNode->Attribute("Password"); XMLElement* genderNode=userNode->FirstChildElement("Gender"); user->gender=atoi(genderNode->GetText()); XMLElement* mobileNode=userNode->FirstChildElement("Mobile"); user->mobile=mobileNode->GetText(); XMLElement* emailNode=userNode->FirstChildElement("Email"); user->email=emailNode->GetText(); return user; } return NULL; }
|
3.4修改xml文件的指定节点
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
|
bool updateUser(const char* xmlPath,User* user) { XMLDocument doc; if(doc.LoadFile(xmlPath)!=0) { cout<<"load xml file failed"<<endl; return false; } XMLElement* root=doc.RootElement(); XMLElement* userNode=queryUserNodeByName(root,user->userName);
if(userNode!=NULL) { if(user->password!=userNode->Attribute("Password")) { userNode->SetAttribute("Password",user->password.c_str()); } XMLElement* genderNode=userNode->FirstChildElement("Gender"); if(user->gender!=atoi(genderNode->GetText())) { genderNode->SetText(itoa(user->gender).c_str()); } XMLElement* mobileNode=userNode->FirstChildElement("Mobile"); if(user->mobile!=mobileNode->GetText()) { mobileNode->SetText(user->mobile.c_str()); } XMLElement* emailNode=userNode->FirstChildElement("Email"); if(user->email!=emailNode->GetText()) { emailNode->SetText(user->email.c_str()); } if(doc.SaveFile(xmlPath)==0) return true; } return false; }
|
验证代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| int main(int argc,char* argv[]) { User user("lvlv","00001111",0,"13995648666","1586666@qq.com"); if(updateUser("./user.xml",&user)) cout<<"update successfully"<<endl; else cout<<"update failed"<<endl; return 0; } 修改结果:
<?xml version="1.0" encoding="UTF-8" standalone="no"?> <DBUSER> <User Name="lvlv" Password="00001111"> <Gender>0</Gender> <Mobile>13995648666</Mobile> <Email>1586666@qq.com</Email> </User> </DBUSER>
|
3.5删除xml文件的指定节点的信息
//function:删除指定节点内容
//param:xmlPath:xml文件路径;userName:用户名称
//return:bool
bool deleteUserByName(const char* xmlPath,const string& userName)
{
XMLDocument doc;
if(doc.LoadFile(xmlPath)!=0)
{
cout<<”load xml file failed”<<endl;
return false;
}
XMLElement* root=doc.RootElement();
//doc.DeleteNode(root);//删除xml所有节点
XMLElement* userNode=queryUserNodeByName(root,userName);
if(userNode!=NULL)
{
userNode->DeleteAttribute(“Password”);//删除属性
XMLElement* emailNode=userNode->FirstChildElement(“Email”);
userNode->DeleteChild(emailNode); //删除指定节点
//userNode->DeleteChildren();//删除节点的所有孩子节点
if(doc.SaveFile(xmlPath)==0)
return true;
}
return false;
}
1 2 3 4 5 6 7 8 9 10 11 12
| 验证代码: ```C++ int main(int argc,char* argv[]) { //删除用户某些信息 if(deleteUserByName("./user.xml","lvlv")) cout<<"delete successfully"<<endl; else cout<<"delete failed"<<endl; return 0; }
|
删除结果:
1 2 3 4 5 6 7
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> <DBUSER> <User Name="lvlv"> <Gender>10</Gender> <Mobile>13995648666</Mobile> </User> </DBUSER>
|
4.其它常见用例
4.1获取xml文件申明
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
|
bool getXMLDeclaration(const char* xmlPath,string& strDecl) { XMLDocument doc; if(doc.LoadFile(xmlPath)!=0) { cout<<"load xml file failed"<<endl; return false; } XMLNode* decl=doc.FirstChild(); if (NULL!=decl) { XMLDeclaration* declaration =decl->ToDeclaration(); if (NULL!=declaration) { strDecl = declaration->Value(); return true; } } return false; }
|
验证代码:
1 2 3 4 5 6 7 8 9 10
| int main(int argc,char* argv[]) { string strDecl; if(getXMLDeclaration("./user.xml",strDecl)) { cout<<"declaration:"<<strDecl<<endl; } return 0; }
|
验证结果:
1
| declaration:xml version="1.0" encoding="UTF-8" standalone="no"
|
4.2打印xml文件至标准输出
1 2 3 4 5 6 7 8 9 10 11 12
|
void print(const char* xmlPath) { XMLDocument doc; if(doc.LoadFile("./user.xml")!=0) { cout<<"load xml file failed"<<endl; return; } doc.Print(); }
|
4.3 xml文件内容输出至内存
1 2 3
| XMLPrinter printer; doc.Print( &printer );
|
4.4如何解析xml格式的字符串信息,不是文件中的
采用tinyXML中的TiXmlDocument类的Parse方法即可。例如:
1 2 3 4 5 6
| const char * xmlString = "<?xml version=\"1.0\" standalone=no>\n<!– Our to do list data –>\n<ToDo>\n<Item priority=\"1\"> <bold>Toy store!</bold>\n</Item>\n<Item priority=\"2\"> Do bills</Item>\n</ToDo> "; XMLDocument *doc = new XMLDocument(); doc->Parse(xmlString); XMLElement * rootElement = doc->RootElement(); const char * rootName = rootElement->Value();
|