xml简单文档
<root>
<abin>newNode1 content</abin>
<wangzhenongyao>newNode2 content</wangzhenongyao>
<xiaoshui>newNode3 content</xiaoshui>
<nodebin attribute="yes">NODE CONTENT</nodebin>
<son/>
</root>
创建
#include<stdio.h>
#include<libxml/parser.h>
#include<libxml/tree.h>
#include <iostream>
using namespace std;
/*
xmlNewDoc 创建一个文档指针doc
xmlNewNode 创建一个节点指针 root_node
xmlDocSetRootElement 将root_node设置为doc的根节点
给root_node添加一系列的子节点,并设置子节点的内容和属性
用xmlSaveFile 将xml文档存入文件
用xmlFreeDoc函数关闭指针
*/
int main()
{
//创建文档和节点指针
xmlDocPtr doc = xmlNewDoc(BAD_CAST"1.0");
if (NULL == doc)
{
printf("Fail to create new xml\n");
return -1;
}
xmlNodePtr root_node = xmlNewNode(NULL,BAD_CAST"root");
//设置根节点
xmlDocSetRootElement(doc,root_node);
//在根节点中直接创建节点
xmlNewTextChild(root_node,NULL,BAD_CAST"abin",BAD_CAST"newNode1 content");
xmlNewTextChild(root_node,NULL,BAD_CAST"wangzhenongyao",BAD_CAST"newNode2 content");
xmlNewTextChild(root_node,NULL,BAD_CAST"xiaoshui",BAD_CAST"newNode3 content");
//创建一个节点,设置其内容和属性,然后加入根节点
xmlNodePtr node = xmlNewNode(NULL,BAD_CAST"nodebin");//xmlNewNode 创建新节点
xmlNodePtr content = xmlNewText(BAD_CAST"NODE CONTENT");//xmlNewText 生成文本节点
xmlAddChild(root_node,node);//给指定的节点添加节点
xmlAddChild(node,content);
xmlNewProp(node,BAD_CAST"attribute",BAD_CAST"yes");//创建新节点属性
#if 1
//创建一个儿子和孙子节点
node = xmlNewNode(NULL,BAD_CAST"son");
xmlAddChild(root_node,node);
xmlNodePtr grandson = xmlNewNode(NULL,BAD_CAST"grandson");
xmlAddChild(grandson,xmlNewText(BAD_CAST"this is a grandson node"));
#endif
//存储xml文档
int nRel = xmlSaveFile("Createdxml.xml",doc);
if(nRel != -1)
{
cout<<"一个xml文档被创建,写入"<<nRel<<"个字节"<<endl;
}
//释放文档内节点动态申请的内存
xmlFreeDoc(doc);
return 0;
}
解析
#include<libxml/parser.h>
#include<libxml/tree.h>
#include<stdio.h>
#include<iostream>
using namespace std;
/***
解析文档
xmlReadFile 函数读出一个文档指针doc
xmlDocGetRootElement函数得到根节点
curNode->xmlChildrenNode就是根节点的子节点集合
轮询子节点集合,找到所需要的节点,用xmlNodeGetContent取出内容
用xmlHasProp查找含有某个属性的节点
取出该节点的属性集合,用xmlGetProp取出属性值
用xmlFreeDoc函数关闭文档指针,并清除文档中的所有节点动态申请的内存
节点的内容是其子节点(对于abin来说,content可以说是其子节点),所以我们需要使用xmlNodeGetContent来取出内容
***/
int main(int argc,char *argv[])
{
xmlDocPtr doc; //定义解析文档指针
xmlNodePtr curNode;//定义节点指针(在各个节点间移动)
xmlChar *szKey; //临时字符串变量
char *szDocName;
if(argc <= 1)
{
printf("Usage:%s DocName\n",argv[0]);
return -1;
}
/*
xmlDocPtr xmlReadFile (const char * filename,
const char * encoding,
filename: 目标文件的路径名或URL
encoding: 目标文件的编码方式,可以为NULL
options: 枚举变量xmlParserOption中的一个值或多个值的组合,可以为0。
返回值: XML文档的树结构。解析失败返回NULL
options的常用取值:
XML_PARSE_RECOVER = 1<<0, /* recover on errors */ 尝试修复错误语法
XML_PARSE_NOERROR = 1<<5, /* suppress error reports */ 不输出错误日志
XML_PARSE_NOWARNING = 1<<6, /* suppress warning reports */ 不输出警告日志
XML_PARSE_PEDANTIC = 1<<7, /* pedantic error reporting */ 输出详细的错误日志
XML_PARSE_NOERROR和XML_PARSE_NOWARNING的功能是“当运行命令行程序时不输出错误/警告日志”。
*/
szDocName = argv[1];
doc = xmlReadFile(szDocName,"GB2312",XML_PARSE_RECOVER);
if(NULL == doc)
{
fprintf(stderr,"Document not parsed successfully");
return -1;
}
//确定文档根元素
curNode = xmlDocGetRootElement(doc);
//检查确认当前文档中包含的内容
if(NULL == curNode)
{
fprintf(stderr,"empty document");
return -1;
}
//确认文档正确的类型."root"是这个示例中使用的文档的根类类型
if(xmlStrcmp(curNode->name,BAD_CAST"root"))
{
fprintf(stderr,"document of the wrong type,root node != root");
xmlFreeDoc(doc);
return -1;
}
curNode = curNode->xmlChildrenNode;//xmlChildrenNode 节点的子节点
xmlNodePtr propNodePtr = curNode;
while(NULL != curNode)
{
if((!xmlStrcmp(curNode->name,(const xmlChar*)"abin")))
{
szKey = xmlNodeGetContent(curNode);
printf("newNodeFFF:%s\n",szKey);
xmlFree(szKey);
}
//查找带有属性attribute的节点
if(xmlHasProp(curNode,BAD_CAST"attribute"))
{
propNodePtr = curNode;
}
curNode = curNode->next;//curNode->next 兄弟节点
}
//查找属性
xmlAttrPtr attPtr = propNodePtr->properties;//属性集是链表
while(NULL == attPtr)
{
if(!xmlStrcmp(attPtr->name,BAD_CAST"attribute"))
{
xmlChar *szAttr = xmlGetProp(propNodePtr,BAD_CAST"attribute");
cout<<"get attribute bin = "<<szAttr<<endl;
xmlFree(szAttr);
}
attPtr = attPtr->next;
}
xmlFreeDoc(doc);
return 0;
}
XPath的七总类型的节点:元素,属性,文本,命名空间,处理指令,注释,文档节点(根节点)