• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            C++中使用Expat解析XML

            使用expat的原因很多,主要還是因為expat更靈活。習慣了TinyXML,一開始不太習慣expat,分析一下,其實很容易上手的。

                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,最后一個素組元素為空指針,奇數指針對應屬性名稱,偶數指針對應屬性值(字符串格式)。可以在一個循環中處理多個屬性,當遇到空指針時,表示沒有更多屬性了。

                好啦,先看sample吧:

                #include <stdio.h>

                #include "expat.h"

                #pragma warning(disable:4996)

                #define XML_FMT_INT_MOD "l"

                static void XMLCALL startElement(void *userData, const char *name, const char **atts)

                {

                int i;

                int *depthPtr = (int *)userData;

                for (i = 0; i < *depthPtr; i++)

                printf(" ");

                printf(name);

                *depthPtr += 1;

                for(i=0;atts[i]!=0;i+=2)

                {

                printf(" %s=%s",atts[i],atts[i+1]);

                }

                printf("\n");

                }

                static void XMLCALL endElement(void *userData, const char *name)

                {

                int *depthPtr = (int *)userData;

                *depthPtr -= 1;

                }

                int main(int argc, char *argv[])

                {

                char buf[BUFSIZ];  XML_Parser parser = XML_ParserCreate(NULL);

                int done;  int depth = 0;

                XML_SetUserData(parser, &depth);

                XML_SetElementHandler(parser, startElement, endElement);

                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;

                }

                4.其他ElementHanlder

                expat還可以設置CData,Comment的handler,另外一些函數本人還沒使用過,涉及到更多的xml標準的知識,如果需要,可以參考官方的手冊。

            posted on 2012-02-26 13:31 一葉草 閱讀(2048) 評論(0)  編輯 收藏 引用
            亚洲欧美日韩精品久久亚洲区 | 国产精品欧美久久久久天天影视| 99久久久国产精品免费无卡顿 | 久久久久AV综合网成人| 久久本道伊人久久| 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 91精品婷婷国产综合久久 | AA级片免费看视频久久| 欧美日韩成人精品久久久免费看| 国产A级毛片久久久精品毛片| 国内精品久久人妻互换| 欧美久久久久久精选9999| 97精品伊人久久久大香线蕉| 久久99国产精品二区不卡| 亚洲AⅤ优女AV综合久久久| 99精品久久精品| 三级三级久久三级久久| 伊人久久综在合线亚洲2019| 无遮挡粉嫩小泬久久久久久久| 7国产欧美日韩综合天堂中文久久久久 | 亚洲va久久久久| 国内精品欧美久久精品| 国产Av激情久久无码天堂| 久久精品日日躁夜夜躁欧美| 久久久久国色AV免费观看| 久久香蕉国产线看观看乱码| 亚洲国产精品18久久久久久| 欧美日韩精品久久久免费观看| 91久久精品电影| 一本伊大人香蕉久久网手机| AV色综合久久天堂AV色综合在| 久久精品无码专区免费青青| 亚洲中文字幕无码久久2020 | 91视频国产91久久久| 无码超乳爆乳中文字幕久久| 国产精品中文久久久久久久| 午夜视频久久久久一区 | 亚洲色大成网站WWW久久九九| 中文字幕亚洲综合久久菠萝蜜| 久久成人18免费网站| 精品久久久久久久久久久久久久久|