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

            桃源谷

            心靈的旅行

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

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

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

            隱鋒同學的blog上有關于libxml2的一篇文章,正好最近要使用這個庫來處理xml文件。
            不過在測試時我們發現用文章里F. 添加屬性例程代碼 時,添加的keyword結點后面沒有回車,
            跟后面的結點擠在一行了,不是很好看。
            例如,有以下的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節點的最后插入一個keyword的子結點后的,
            該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>

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

            通過設定 xmlKeepBlanksDefault(0) 以及 xmlSaveFormatFile(...)的format參數設置成1,都無法實現
            在新追加的結點后面添加回車換行。
            www.xmlsoft.org的官方網站的maillist里關于這方面的信息非常少。但是,對我幫助最大還是
            http://mail.gnome.org/archives/xml/2007-May/msg00043.html 這個問題里的例子代碼,里面在設置
            屬性的時候用的xmlReadFile函數,而且options參數設定的是XML_PARSE_NOBLANKS。

            于是,我們用xmlReadFile(...),把它的options參數設定成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參數設定成XML_PARSE_NOBLANKS;否則的話是不會在結點后面添加回車的。
            修正2:把xmlSaveFormatFileformat參數修改成1,否則在使用xmlReadFile打開的xml文件時,在生成的xml文件里是會把所有的結點都放到一行里顯示。
            另外:xmlKeepBlanksDefault(0) 除了在讀入xml文件時忽略空白之外,還會在寫出xml文件時在每行前面放置縮進(indent)。如果使用xmlKeepBlanksDefault(1) 則你會發現每行前面的縮進就沒有了,但不會影響回車換行。

            //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
            額外話題: 更新結點的值得時候segement fault錯誤
            下面的代碼是更新XML文件里的某些結點元素的值的簡單的例子。
             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函數xmlNodeSetContent
            處發生段錯誤,但不是100%發生。
            只有該結點在原來值是某些字符串的時候會發生該錯誤,比如說,
            原來的值是"zo"的時候就會讓程序崩潰。
            閱讀了libxml2的源代碼發現,xmlNodeSetContent函數,在把結點值
            設置成新的字符串之前會調用xmlFree(cur->content)來釋放掉原來
            字符串緩沖區的內存。
            xmlNodeSetContent函數的代碼片斷:
             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        }
            在上面代碼里,如果結點值得字符串如果在libxml2的字典緩沖區(cur->doc->dict)里,
            就把該字符串釋放掉。而原來的字符串"zo"恰好在它的字典緩沖里,那這樣傳遞到
            xmlFree函數里的地址是沖區的一部分而不是緩沖區的首地址的話,free函數當然
            會死掉了。如果換成其他的字符串就沒有任何問題。

            但是,令人不解的是在libxml2的另一部分代碼里,刪除節點的程序去不是這樣做。
            例如,在一個結點被刪除后,通常會使用xmlFreeDoc函數來釋放該結點,恰好在這段
            代碼里卻是判斷如果該字符串不再字典緩沖區才去釋放它,也就是調用宏DICT_FREE
            來完成釋放工作,這兒正好與前面的相反,很難理解為什么會產生矛盾。
            宏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函數的bug,不過在maillist和bugzilla也沒有翻到關于它
            的任何說明。

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

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

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

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

            Feedback

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

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

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

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

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

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

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

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

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

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

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

            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,而是 參數調用有先后的問題.
            后面的問題

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

            2012-01-12 12:55 by wzhang
            冒昧的問個問題:在libxml中可以獲取<![CDATA[ ]]中的內容嗎?
            我的個人簡歷第一頁 我的個人簡歷第二頁
            国产亚洲精久久久久久无码| 久久国产精品成人影院| 久久久久久久人妻无码中文字幕爆| 久久国产精品-国产精品| 97久久精品人妻人人搡人人玩 | 国产福利电影一区二区三区久久久久成人精品综合 | 欧美大香线蕉线伊人久久| 亚洲va久久久噜噜噜久久 | 青草影院天堂男人久久| 久久国产成人午夜AV影院| 青青草国产97免久久费观看| 国产欧美久久久精品影院| 久久综合九色综合网站| 久久久久亚洲AV无码麻豆| 97久久超碰成人精品网站| 国产精品九九九久久九九| 青青青伊人色综合久久| 久久久久久A亚洲欧洲AV冫| 亚洲中文精品久久久久久不卡 | 久久久久久综合一区中文字幕 | 久久综合噜噜激激的五月天| 久久99精品国产自在现线小黄鸭| 1000部精品久久久久久久久| 国产成人无码精品久久久免费| 久久国产乱子伦精品免费午夜| 婷婷久久综合九色综合绿巨人| 伊人久久大香线蕉av一区| 久久精品国产亚洲av水果派| 99久久国产亚洲高清观看2024 | 久久亚洲国产精品一区二区| 国产精品综合久久第一页| 久久99久久99精品免视看动漫| 国产毛片久久久久久国产毛片 | 亚洲欧美另类日本久久国产真实乱对白 | 伊人色综合久久天天网| 色偷偷久久一区二区三区| 国产精品99久久久久久猫咪 | 久久AV无码精品人妻糸列| 91久久精品国产成人久久| 午夜天堂精品久久久久| 久久精品夜色噜噜亚洲A∨|