3.3 重要操作
3.3.1 创建XML文档
创建一个xml文档流程如下:
l 用xmlNewDoc函数创建一个文档指针doc;
l 用xmlNewNode函数创建一个节点指针root_node;
l 用xmlDocSetRootElement将root_node设置为doc的根结点;
l 给root_node新增一系列的子节点,并设置子节点的内容和属性;
l 用xmlSaveFile将xml文档存入文件;
l 用xmlFreeDoc函数关闭文档指针,并断根本文档中所有节点动态声请的内存
示例代码:
#include <stdio.h> #include <libxml/parser.h> #include <libxml/tree.h> int main(int argc,char **argv) { xmlDocPtr pdoc = NULL; xmlNodePtr proot_node = NULL,pnode = NULL,pnode1 = NULL; // 创造一个新文档并设置要root节点
// 在XML文档中有且只有一个root节点
pdoc = xmlNewDoc( BAD_CAST "1.0");
proot_node = xmlNewNode(NULL,BAD_CAST "plist");
xmlNewProp(proot_node,BAD_CAST "version",BAD_CAST "1.0");
xmlDocSetRootElement(pdoc,proot_node);
pnode = xmlNewNode( NULL,BAD_CAST "dict" );
// 创造root节点的子节点
xmlNewChild(pnode,NULL,BAD_CAST "key",BAD_CAST "info");
pnode1 = xmlNewNode( NULL,BAD_CAST "dict" );
//保存文档
xmlSaveFormatFileEnc(argc > 1 ? argv[1] : "-",pdoc,"UTF-8",1);
//释放文档指针
xmlFreeDoc(pdoc);
xmlCleanupParser();
xmlMemoryDump(); //debug memory for regression tests
return(0);
} 涉及相关函数:
xmlNewDoc,xmlNewNode,xmlDocSetRootElement,xmlSaveFile,xmlFreeDoc。
3.3.2 解释文档
解析一个xml文档,从其中取出想要的信息,例如节点中包含的书契,或者某个节点的属性,其流程如下:
l 用xmlReadFile函数读出一个文档指针doc;
l 用xmlDocGetRootElement函数获得根节点curNode;
l curNode->xmlChildrenNode就是根节点的子节点集合;
l 轮询子节点集合,找到所需的节点,用xmlNodeGetContent取出其内容;
l 用xmlHasProp查找含有某个属性的节点;
l 取出该节点的属性集合,用xmlGetProp取出其属性值;
l 用xmlFreeDoc函数关闭文档指针,并断根本文档中所有节点动态声请的内存。
注意:节点列表的指针依然是xmlNodePtr,属性列表的指针也是xmlAttrPtr,并没有xmlNodeList或者xmlAttrList这样的类型。看作列表的时候使用它们的next和prev链表指针来进行轮询。只有在Xpath中有xmlNodeSet这种类型,其使用方法前边已经先容了。
示例代码:
int main(int argc,char **argv) { xmlDocPtr pdoc = NULL; xmlNodePtr proot=NULL,电脑urnode = NULL; char *psfilename; if (argc < 1) { printf("error: Usage: %s filename/n",argv[0]); exit(1); } psfilename = argv[1]; pdoc = xmlReadFile(psfilename,XML_PARSE_RECOVER); if(NULL == pdoc) { printf("error: open file %s",psfilename); exit(1); } 电脑urnode = xmlDocGetRootElement(pdoc); if(NULL == 电脑urnode) { printf("error: empty document %s",psfilename); exit(1); } if(xmlStrcmp(电脑urnode->name,BAD_CAST "plist") != 0) { printf("error document " ); exit(1); } if(xmlHasProp(电脑urnode,BAD_CAST "version")) { xmlChar *szAttr = xmlGetProp(电脑urnode,BAD_CAST "version"); printf("get version : %s /n",szAttr); } xmlFreeDoc(pdoc); xmlCleanupParser(); return 0; } 涉及函数:
xmlParseFile,xmlReadFile,xmlDocGetRootElement,xmlHasProp,xmlGetProp。其中xmlParseFile和xmlReadFile都具有打开XML文件的功能,xmlParseFile函数以默认方式读入一个UTF-8格局的文档,并返回文档指针。xmlReadFile函数读入一个带有某种编码的xml文档,并返回文档指针。 3.3.3 修改XML
有了上边的基础,修改xml文档的内容就很简单了。起首打开一个已经存在的xml文档,顺着根结点找到需要新增、删去、修改之处,调用相应的xml函数对节点进行增、删、改操作。
示例代码:
#include <libxml/parser.h> #include <iostream.h> int main(int argc,char* argv[]) { xmlDocPtr doc; //界说解析文档指针
xmlNodePtr curNode; //界说结点指针你需要它为了在各个结点间移动)
char *szDocName;
if (argc <= 1)
{
printf("Usage: %s docname/n",argv[0]);
return(0);
}
szDocName = argv[1];
doc = xmlReadFile(szDocName,"GB2312",XML_PARSE_RECOVER); //解析文件
if (NULL == doc)
{
fprintf(stderr,"Document not parsed successfully. /n");
return -1;
}
curNode = xmlDocGetRootElement(doc);
if (NULL == curNode)
{
fprintf(stderr,"empty document/n");
xmlFreeDoc(doc);
return -1;
}
curNode = curNode->children;
while (NULL != curNode)
{
//删去newNode1
if (!xmlStrcmp(curNode->name,BAD_CAST "newNode1"))
{
xmlNodePtr tempNode;
tempNode = curNode->next;
xmlUnlinkNode(curNode);
xmlFreeNode(curNode);
curNode = tempNode;
continue;
}
if (!xmlStrcmp(curNode->name,BAD_CAST "node2"))
{
xmlSetProp(curNode,BAD_CAST "attribute",BAD_CAST "no");
}
if (!xmlStrcmp(curNode->name,BAD_CAST "newNode2"))
{
xmlNodeSetContent(curNode,BAD_CAST "content changed");
}
if (!xmlStrcmp(curNode->name,BAD_CAST "newNode3"))
{
xmlNewProp(curNode,BAD_CAST "newAttr",BAD_CAST "YES");
}
//增加一个子节点
if (!xmlStrcmp(curNode->name,BAD_CAST "son"))
{
xmlNewTextChild(curNode,BAD_CAST "newGrandSon",BAD_CAST "new content");
}
curNode = curNode->next;