2. make && make install
1.回调函数
以下案例解析xml文件中的elment,attribute和text。expat使用回调方式返回xml数据,解析器解析到一个element及其内部属性后,将调用事先设置好的函数,同样,当element结束和text结束后,也会分别调用对应的函数。
2.如何处理数据之间的包含关系
典型的方式是定义三个函数分别处理elment开始(含属性)、element结束和文本内容。回调函数的第一个参数是自定义的,通常用于存储XML文档的上下文信息,用XML_SetUserData可以设置这个参数,下例中传递一个整数指针,以便在每次回调时能知道该元素是第几层元素。
该参数也可以是一个栈对象的地址,开始一个元素时,将新元素对应的数据压入堆栈,处理下一级元素时,新元素是栈顶元素在子元素,然后处理完了继续把该元素压入堆栈,继续下一级新的子元素。当元素结束后,需要出栈,以便解析下个兄弟元素程时能取到父节点。
好啦,基本应用还是很简单的,实际上Expat的API函数不多。
3.如何处理属性
属性通过ElementHandler回调函数传入,这里有一个char** atts就是属性,这是一个字符指针数组,如果有N个属性,数组大小就是2*N+1,最后一个素组元素为空指针,奇数指针对应属性名称,偶数指针对应属性值(字符串格式)。可以在一个循环中处理多个属性,当遇到空指针时,表示没有更多属性了。
demo
#include <stdio.h>
#include "expat.h"
#pragma warning(disable:4996)
#define XML_FMT_INT_MOD "l"
int start_pos = 0;
static void XMLCALL startElement(void *userData, const char *name, const char **atts)
{
int i;
int *depthPtr = (int *)userData;
*depthPtr += 1;
printf(" %s start", name);
printf("\n");
}
static void XMLCALL endElement(void *userData, const char *name)
{
printf(" %s end", name);
printf("\n");
int *depthPtr = (int *)userData;
*depthPtr -= 1;
}
static void ValueElement(void *userData, const XML_Char *value, int len){
int *depthPtr = (int *)userData;
printf(" %s %d data", value, len);
printf("\n");
}
int main(int argc, char *argv[])
{
char buf[BUFSIZ];
XML_Parser parser = XML_ParserCreate(NULL);
int done; int depth = 0;
XML_SetUserData(parser, &depth); //设置userdata的数据变量
XML_SetElementHandler(parser, startElement, endElement); //设置节点开始 和 结束的回调函数
XML_SetCharacterDataHandler(parser, ValueElement); //设置节点内数据的回调函数
FILE* pFile= argc<2 ?stdin : fopen(argv[1],"rb");
do
{ int len = (int)fread(buf, 1, sizeof(buf), pFile);
done = len < sizeof(buf);
if (XML_Parse(parser, buf, len, done) == XML_STATUS_ERROR)
{
fprintf(stderr,"%s at line %" XML_FMT_INT_MOD "u\n",
XML_ErrorString(XML_GetErrorCode(parser)),
XML_GetCurrentLineNumber(parser));
return 1;
}
}
while (!done);
XML_ParserFree(parser);
fclose(pFile);
return 0;
}