• <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>

            桃源谷

            心靈的旅行

            人生就是一場旅行,不在乎旅行的目的地,在乎的是沿途的風(fēng)景和看風(fēng)景的心情 !
            posts - 32, comments - 42, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            在xml里追加結(jié)點時添加回車(libxml2)

            Posted on 2007-11-30 13:44 lymons 閱讀(10150) 評論(8)  編輯 收藏 引用 所屬分類: C++CUnix/Linux
            From 2008精選

            隱鋒同學(xué)的blog上有關(guān)于libxml2的一篇文章,正好最近要使用這個庫來處理xml文件。
            不過在測試時我們發(fā)現(xiàn)用文章里F. 添加屬性例程代碼 時,添加的keyword結(jié)點后面沒有回車,
            跟后面的結(jié)點擠在一行了,不是很好看。
            例如,有以下的xml例子文件
             1<?xml version="1.0"?>
             2<BODY>
             3  <filesystem>
             4    <filesystemKeyData>
             5      <filesystemName>Ext3</filesystemName>
             6      <versionNumber>123</versionNumber>
             7      <option>good</option>
             8    </filesystemKeyData>
             9    <timestampSec>456</timestampSec>
            10    <status>heasjdkfjaskdfjsk</status>
            11  </filesystem>
            12  <filesystem>
            13    <filesystemKeyData>
            14      <filesystemName>Ext3</filesystemName>
            15      <versionNumber>123</versionNumber>
            16      <option>good</option>
            17    </filesystemKeyData>
            18    <timestampSec>456</timestampSec>
            19    <status>heasjdkfjaskdfjsk</status>
            20  </filesystem>
            21</BODY>


            例如,使用該文章例子中的代碼在上面的filesystem節(jié)點的最后插入一個keyword的子結(jié)點后的,
            該xml文件的表示如下:

             1<?xml version="1.0"?>
             2<BODY>
             3  <filesystem>
             4    <filesystemKeyData>
             5      <filesystemName>Ext3</filesystemName>
             6      <versionNumber>123</versionNumber>
             7      <option>good</option>
             8    </filesystemKeyData>
             9    <timestampSec>456</timestampSec>
            10   <status>heasjdkfjaskdfjsk</status>
            11   <keyword1>hello</keyword1><keyword2>hello</keyword2><keyword3>hello</keyword3></filesystem>
            12  <filesystem>
            13    <filesystemKeyData>
            14      <filesystemName>Ext3</filesystemName>
            15      <versionNumber>123</versionNumber>
            16      <option>good</option>
            17    </filesystemKeyData>
            18    <timestampSec>456</timestampSec>
            19    <status>heasjdkfjaskdfjsk</status>
            20    <keyword1>hello</keyword1><keyword2>hello</keyword2><keyword3>hello</keyword3></filesystem>
            21</BODY>

            你會發(fā)現(xiàn)keyword和/filesystem像下面那樣被擠在一起了,這并不是我們想要的.
            <keyword1>hello</keyword1><keyword2>hello</keyword2><keyword3>hello</keyword3></filesystem>

            通過設(shè)定 xmlKeepBlanksDefault(0) 以及 xmlSaveFormatFile(...)的format參數(shù)設(shè)置成1,都無法實現(xiàn)
            在新追加的結(jié)點后面添加回車換行。
            www.xmlsoft.org的官方網(wǎng)站的maillist里關(guān)于這方面的信息非常少。但是,對我?guī)椭畲筮€是
            http://mail.gnome.org/archives/xml/2007-May/msg00043.html 這個問題里的例子代碼,里面在設(shè)置
            屬性的時候用的xmlReadFile函數(shù),而且options參數(shù)設(shè)定的是XML_PARSE_NOBLANKS。

            于是,我們用xmlReadFile(...),把它的options參數(shù)設(shè)定成XML_PARSE_NOBLANKS后,就可以自動添加
            回車了。

            那,重新修正了的例子程序是如下那樣,里面只修改了兩條語句。

             1#include <stdio.h>
             2#include <string.h>
             3#include <stdlib.h>
             4#include <libxml/xmlmemory.h>
             5#include <libxml/parser.h>
             6void
             7parseStory (xmlDocPtr doc, xmlNodePtr cur, char *keyword)
             8{
             9   xmlNewTextChild (cur, NULL, "keyword1", keyword);
            10  xmlNewTextChild (cur, NULL, "keyword2", keyword);
            11  xmlNewTextChild (cur, NULL, "keyword3", keyword);
            12  return;
            13}

            14
            15xmlDocPtr
            16parseDoc (char *docname, char *keyword)
            17{
            18  xmlDocPtr doc;
            19  xmlNodePtr cur;
            20  //doc = xmlParseFile (docname);
            21  doc = xmlReadFile(docname, NULL, XML_PARSE_NOBLANKS);
            //讀取xml文件時忽略空格
            22  if (doc == NULL)
            23  {
            24      fprintf (stderr, "Document not parsed successfully. \n");
            25      return (NULL);
            26  }

            27  cur = xmlDocGetRootElement (doc);
            28  if (cur == NULL)
            29  {
            30      fprintf (stderr, "empty document\n");
            31      xmlFreeDoc (doc);
            32      return (NULL);
            33  }

            34  if (xmlStrcmp (cur->name, (const xmlChar *"BODY"))
            35  {
            36      fprintf (stderr, "document of the wrong type, root node != story\n");
            37      xmlFreeDoc (doc);
            38      return (NULL);
            39  }

            40  cur = cur->xmlChildrenNode;
            41  while (cur != NULL)
            42  {
            43      if ((!xmlStrcmp (cur->name, (const xmlChar *"filesystem")))
            44      {
            45         parseStory (doc, cur, keyword);
            46      }

            47      cur = cur->next;
            48  }

            49  return (doc);
            50}

            51
            52int
            53main (int argc, char **argv)
            54{
            55  char *docname;
            56  char *keyword;
            57  xmlDocPtr doc;
            58  if (argc <= 2)
            59  {
            60      printf ("Usage: %s docname, keyword\n", argv[0]);
            61      return (0);
            62  }

            63  docname = argv[1];
            64  keyword = argv[2];
            65  doc = parseDoc (docname, keyword);
            66  if (doc != NULL)
            67  {
            68      //xmlSaveFormatFile (docname, doc, 0);
            69      xmlSaveFormatFile (docname, doc, 1);
            70      xmlFreeDoc (doc);
            71  }

            72  return (1);
            73}

            74
            修正1:是把xmlParseFile替換成xmlReadFile,并且是options參數(shù)設(shè)定成XML_PARSE_NOBLANKS;否則的話是不會在結(jié)點后面添加回車的。
            修正2:把xmlSaveFormatFileformat參數(shù)修改成1,否則在使用xmlReadFile打開的xml文件時,在生成的xml文件里是會把所有的結(jié)點都放到一行里顯示。
            另外:xmlKeepBlanksDefault(0) 除了在讀入xml文件時忽略空白之外,還會在寫出xml文件時在每行前面放置縮進(indent)。如果使用xmlKeepBlanksDefault(1) 則你會發(fā)現(xiàn)每行前面的縮進就沒有了,但不會影響回車換行。

            //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
            額外話題: 更新結(jié)點的值得時候segement fault錯誤
            下面的代碼是更新XML文件里的某些結(jié)點元素的值的簡單的例子。
             1    xmlNodePtr element;
             2    // 
             3    xmlNodePtr childrenNodePtr = element->children;
             4    while(childrenNodePtr != NULL)
             5    {
             6        if(childrenNodePtr->type == XML_TEXT_NODE)
             7        {
             8            xmlNodeSetContent(childrenNodePtr, (const xmlChar*)"world");
             9            return NORMAL_RET;
            10        }

            11        childrenNodePtr = childrenNodePtr->next;
            12    }

            運行該段代碼,有時候會在使用libxml2的API函數(shù)xmlNodeSetContent
            處發(fā)生段錯誤,但不是100%發(fā)生。
            只有該結(jié)點在原來值是某些字符串的時候會發(fā)生該錯誤,比如說,
            原來的值是"zo"的時候就會讓程序崩潰。
            閱讀了libxml2的源代碼發(fā)現(xiàn),xmlNodeSetContent函數(shù),在把結(jié)點值
            設(shè)置成新的字符串之前會調(diào)用xmlFree(cur->content)來釋放掉原來
            字符串緩沖區(qū)的內(nèi)存。
            xmlNodeSetContent函數(shù)的代碼片斷:
             1switch (cur->type) {
             2        case XML_DOCUMENT_FRAG_NODE:
             3        case XML_ELEMENT_NODE:
             4        case XML_ATTRIBUTE_NODE:
             5        if (cur->children != NULL) xmlFreeNodeList(cur->children);
             6        cur->children = xmlStringGetNodeList(cur->doc, content);
             7        UPDATE_LAST_CHILD_AND_PARENT(cur)
             8        break;
             9        case XML_TEXT_NODE:
            10        case XML_CDATA_SECTION_NODE:
            11        case XML_ENTITY_REF_NODE:
            12        case XML_ENTITY_NODE:
            13        case XML_PI_NODE:
            14        case XML_COMMENT_NODE:
            15        if ((cur->content != NULL) &&
            16            (cur->content != (xmlChar *&(cur->properties))) {
            17            if (!((cur->doc != NULL) && (cur->doc->dict != NULL) &&
            18            (xmlDictOwns(cur->doc->dict, cur->content))))
            19            xmlFree(cur->content);
            20        }
            在上面代碼里,如果結(jié)點值得字符串如果在libxml2的字典緩沖區(qū)(cur->doc->dict)里,
            就把該字符串釋放掉。而原來的字符串"zo"恰好在它的字典緩沖里,那這樣傳遞到
            xmlFree函數(shù)里的地址是沖區(qū)的一部分而不是緩沖區(qū)的首地址的話,free函數(shù)當(dāng)然
            會死掉了。如果換成其他的字符串就沒有任何問題。

            但是,令人不解的是在libxml2的另一部分代碼里,刪除節(jié)點的程序去不是這樣做。
            例如,在一個結(jié)點被刪除后,通常會使用xmlFreeDoc函數(shù)來釋放該結(jié)點,恰好在這段
            代碼里卻是判斷如果該字符串不再字典緩沖區(qū)才去釋放它,也就是調(diào)用宏DICT_FREE
            來完成釋放工作,這兒正好與前面的相反,很難理解為什么會產(chǎn)生矛盾。
            宏DICT_FREE的代碼:
             1/**//**
             2 * DICT_FREE:
             3 * @str:  a string
             4 *
             5 * Free a string if it is not owned by the "dict" dictionnary in the
             6 * current scope
             7 */
             8#define DICT_FREE(str)                        \
             9    if ((str) && ((!dict) ||                 \
            10        (xmlDictOwns(dict, (const xmlChar *)(str)) == 0)))    \
            11        xmlFree((char *)(str));
            上面這段代碼判斷是該字符串如果不在字典緩沖里才去釋放。

            在考慮這是否是xmlNodeSetContent函數(shù)的bug,不過在maillist和bugzilla也沒有翻到關(guān)于它
            的任何說明。

            開發(fā)時間上也不允許去跟libxml2深究它是否是bug,只要采用了迂回策略了。
            想辦法不讓libxml2產(chǎn)生字典緩沖不就可以了嗎。
            通過官方手冊我們可以知道,xmlReadFile函數(shù)可以附加XML_PARSE_NODICT選項
            來避免產(chǎn)生字典緩沖。就像下面這樣:
            doc = xmlReadFile(docname, NULL, XML_PARSE_NOBLANK | XML_PARSE_NODICT);

            這樣的話,最開始的那段程序運行起來就沒有任何問題了。

            隱鋒同學(xué)的文章在這里:http://www.cnblogs.com/coolattt/articles/804112.html

            還有,xmlsoft上還有很多使用libxml2的例子程序,可以參考一下:
            http://xmlsoft.org/examples/index.html

            Feedback

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2009-03-30 09:40 by tomsun
            謝謝你提供的資料!非常感謝,很詳細,謝謝!

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2009-04-02 22:46 by sun2bird
            多謝,對于我目前的一個需求有些幫助.另外,想請教一下,要在xml文件結(jié)尾追加新節(jié)點,怎么處理?

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2009-04-03 10:28 by lymons
            @sun2bird
            你的目的是在根節(jié)點的末尾添加一個子節(jié)點呢? 還是在根節(jié)點之后追加新節(jié)點?
            我所知道的,一般情況下,xml只有一個根節(jié)點. 如果在根節(jié)點之后添入新節(jié)點的話,恐怕有問題,是吧.

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2010-01-10 16:49 by cyz
            看貼后,幫助我解決了一個問題,多謝樓主分享。非常感謝!

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2010-12-23 10:25 by terry wang
            非常感謝,正好要用這個libxml2 寫個xml程序。

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2011-12-17 11:46 by ayanmw
            可以看我的博文
            http://www.cnblogs.com/ayanmw/archive/2011/12/16/2290561.html

            其實 xmlKeepBlanksDefault(0) ;variable .xmlIndentTreeOutput = 1 ;
            添加到 parseFile 或者 readfile 之前就可以了...
            并不是bug,而是 參數(shù)調(diào)用有先后的問題.
            后面的問題

            # re: 在xml里追加結(jié)點時添加回車(libxml2)  回復(fù)  更多評論   

            2012-01-12 12:55 by wzhang
            冒昧的問個問題:在libxml中可以獲取<![CDATA[ ]]中的內(nèi)容嗎?
            我的個人簡歷第一頁 我的個人簡歷第二頁
            久久青青草原综合伊人| 亚洲精品tv久久久久久久久| 久久久无码精品亚洲日韩按摩| 人妻无码αv中文字幕久久琪琪布 人妻无码久久一区二区三区免费 人妻无码中文久久久久专区 | 99久久精品国产一区二区| 国产婷婷成人久久Av免费高清 | 久久精品国产亚洲7777| 一本色道久久88—综合亚洲精品| 99久久精品国产高清一区二区| 久久精品成人影院| 欧美噜噜久久久XXX| 天堂无码久久综合东京热| 国产精品久久久久久影院 | 久久久久18| 久久久久99精品成人片欧美| 久久精品无码一区二区app| 久久国产精品成人影院| 香蕉久久夜色精品国产尤物| 品成人欧美大片久久国产欧美| 伊人色综合久久天天人手人婷 | 97久久天天综合色天天综合色hd| 久久国产影院| 91精品国产91久久久久久| 久久综合给合久久狠狠狠97色69| 亚洲国产成人久久一区久久| 国产99久久久国产精品~~牛| 国产午夜精品久久久久免费视| 国产美女亚洲精品久久久综合| 欧美午夜A∨大片久久| 国产精品日韩欧美久久综合| 色成年激情久久综合| 日本道色综合久久影院| 精品综合久久久久久97超人| 国产成人无码久久久精品一| 色欲久久久天天天综合网精品| 伊人久久大香线蕉av一区| 亚洲级αV无码毛片久久精品| 久久精品国产男包| 午夜精品久久久久久毛片| 99久久精品国产一区二区| 久久久婷婷五月亚洲97号色|