使用tinyxml2将子树从文本插入现有XML文件

时间:2014-09-02 14:37:17

标签: c++ xml insert tinyxml subtree

我不允许创建新标签'tinyxml2',这就是我使用'tinyxml'标签的原因,但我使用的是'tinyxml2'!

我正在尝试将子树元素插入到现有的XML文件中。我的问题是,在运行程序并检查XML文件之后,子树根本不存在于文档中。在原始代码中,我也在加载和保存文件时检查错误,因此这些函数没有问题,它们正常工作。我尝试了一些不同的方法,并使用UserList.NewElement(*name*)添加单个元素 - 函数也可以正常工作。 现在我想从文本变量中插入一个完整的子树......

我的最新方法如​​下(简化而不检查LoadFileSaveFile):

tinyxml2::XMLDocument UserList;
UserList.LoadFile(*Path*);

const char* XMLText = "<user name=\"test-user\" gender=\"male\"><ability description=\"I_can_do_magic\" /></user>";

tinyxml2::XMLDocument TestParse;
TestParse.Parse(XMLText);
tinyxml2::XMLElement* myNewUser = TestParse.RootElement();
UserList.FirstChildElement( "magicians" )->InsertEndChild(myNewUser);
UserList.SaveFile(*Path*);

顺便说一下...... 当我尝试使用XMLText解析tinyxml2::XMLDocument UserList时,保存的XML文件在运行程序后将为空。这意味着在尝试执行此操作时,既不会保存原始XML文档内容,也不会保存新解析的子树。这个事实让我使用了第二个tinyxml2::XMLDocument TestParse。现在保存的XML文件包含它的原始内容,但解析后的子树仍然缺失...非常感谢您提供任何解决方案/帮助/建议。

2 个答案:

答案 0 :(得分:1)

TinyXML-2为存储在XMLDocument中的内存池中的节点(XMLNode)分配内存。这解决了TinyXML-1中存在的内存碎片问题。

副作用是元素无法从一个XMLDocument移动到另一个XMLDocument。它们只能被复制。遗憾的是,TinyXML-2目前不支持深层拷贝(树拷贝),因此无法做到你想要的。 (虽然github网站上要求提供深层复制。)

顺便说一句,我希望你编写的代码断言(在调试模式下)或崩溃,因为myNewUser与UserList位于不同的内存池中。

答案 1 :(得分:1)

我使用TinyXML-2的XMLVisitor编写了深层复制功能。希望这对你有用:

#include <stack>
#include "tinyxml2.h"
using namespace tinyxml2;
class MyXMLVisitor: public XMLVisitor
{
public:
    MyXMLVisitor(XMLDocument *doc)
        : m_doc(doc)
    {
    }

    virtual bool VisitEnter (const XMLElement &el, const XMLAttribute *attr)
    {
        XMLElement *new_el = m_doc->NewElement(el.Name());
        m_elementStack.push(new_el);
        return true;
    }

    virtual bool Visit(const XMLText &txt)
    {
        m_elementStack.top()->SetText(txt.Value());
        return true;
    }

    virtual bool VisitExit (const XMLElement &el)
    {
        XMLElement *top_el = m_elementStack.top();
        m_elementStack.pop();
        if (m_elementStack.empty()) {
            m_element = top_el;
            return false;
        }
        else {
            m_elementStack.top()->InsertEndChild(top_el);
            return true;
        }
    }


    std::stack<XMLElement*> m_elementStack;
    XMLDocument *m_doc;
    XMLElement *m_element;
};


XMLElement* DeepCopyElement(XMLDocument &doc, const XMLElement *el)
{
    MyXMLVisitor my_visitor(&doc);
    el->Accept(&my_visitor);
    return my_visitor.m_element;
}

int main(int argc, char* argv[])
{
    XMLDocument doc;
    doc.LoadFile( "test.xml" );

    XMLElement *modulesElement = doc.FirstChildElement("modules");
    XMLElement *moduleElement = modulesElement->FirstChildElement("module");
    modulesElement->InsertEndChild(DeepCopyElement(doc, moduleElement));

    doc.SaveFile("test_out.xml");
    return 0;
}