• tinyxml学习一


    在TinyXML中,根据XML的各种元素来定义了一些类:
            TiXmlBase:整个TinyXML模型的基类。
            TiXmlAttribute:对应于XML中的元素的属性。
            TiXmlNode:对应于DOM结构中的节点。
            TiXmlComment:对应于XML中的注释。
            TiXmlDeclaration:对应于XML中的申明部分,即<?versiong="1.0" ?>。
            TiXmlDocument:对应于XML的整个文档。
            TiXmlElement:对应于XML的元素。
            TiXmlText:对应于XML的文字部分。
            TiXmlUnknown:对应于XML的未知部分。 
            TiXmlHandler:定义了针对XML的一些操作。

    例如:

    <?xml version="1.0" standalone=no>
    <!– Our to do list data –>
    <ToDo>
    <Item priority="1"> Go to the <bold>Toy store!</bold></Item>
    <Item priority="2"> Do bills</Item>
    </ToDo> 



    整个对象树: 

    TiXmlDocument "demo.xml"
    TiXmlDeclaration "version=’1.0′" "standalone=no"
    TiXmlComment " Our to do list data"
    TiXmlElement "ToDo"
    TiXmlElement "Item" Attribtutes: priority = 1
    TiXmlText "Go to the "
    TiXmlElement "bold"
    TiXmlText "Toy store!"
    TiXmlElement "Item" Attributes: priority=2
    TiXmlText "Do bills" 

    tinyXML中,用FirstChild"名字"时,FirstChild函数的点与要找的点必父子

    句柄

    想要健壮地读取一个XML文档,检查方法调用后的返回值是否为null是很重要的。一种安全的检错实现可能会产生像这样的代码: 

     

    TiXmlElement* root = document.FirstChildElement( "Document" );
    if ( root )
    {
    TiXmlElement* element = root->FirstChildElement( "Element" );
    if ( element )
    {
    TiXmlElement* child = element->FirstChildElement( "Child" );
    if ( child )
    {
    TiXmlElement* child2 = child->NextSiblingElement( "Child" );
    if ( child2 )
    {
    // Finally do something useful. 



    用句柄的话就不会这么冗长了,使用TiXmlHandle类,前面的代码就会变成这样: 

    TiXmlHandle docHandle( &document );
    TiXmlElement* child2 = docHandle.FirstChild( "Document" ).FirstChild( "Element" ).Child( "Child", 1 ).ToElement();
    if ( child2 )
    {
    // do something useful 

    一、读取XML,设置节点文本
    如下XML片段:

    <?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
    <ZXML>
        <ZAPP>
             <VBS_RUNTIME_PARAMS>
                   <BROADCAST_VERSION info="版本">8</BROADCAST_VERSION>
                   <Broadcast>
                    <FileCount info="资源文件个数">69</FileCount>
                    <SOURCE_1>
                        <ID info="图片编号">1</ID>
                        <Version info="图片版本">1</Version>
                        <Path info="图片路径">/mnt/share/1.bmp</Path>
                        <FileMode info="文件处理模式">0</FileMode>
                    </SOURCE_1>
                    <SOURCE_2>
                        <Path info="图片路径">/mnt/share/2.bmp</Path>
                        <ID info="图片编号">2</ID>
                        <Version info="图片版本">1</Version>
                        <FileMode info="文件处理模式">0</FileMode>
                    </SOURCE_2>
                    .
                    </Broadcast>
             </VBS_RUNTIME_PARAMS>
        </ZAPP>
    </ZXML>


    要设置BROADCAST_VERSION节点的值 8为其他值,可参考如下代码(将值加1):
    用ReplaceChild( TiXmlNode* replaceThis, const TiXmlNode& withThis )方法替换

        TiXmlDocument doc("zapp.conf");
        doc.LoadFile();
        TiXmlHandle docHandle( &doc );
    TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChildElement("BROADCAST_VERSION").ToElement();
        TiXmlNode * oldnode =  Broadcast_ver->FirstChild();
        const char *ver = Broadcast_ver->GetText();
        int oldVer = atoi(ver);
        CString newVer;
        newVer.Format("%d",oldVer+1);
        TiXmlText newText(newVer);
        Broadcast_ver->ReplaceChild(oldnode,newText);
        AfxMessageBox(Broadcast_ver->GetText());//输出值
        
    doc.SaveFile();

    二,删除节点,属性值

    RemoveChild( TiXmlNode* removeThis )方法删除父节点的子节点,
    RemoveAttribute( const char * name )方法删除属性值.

    例如删除BROADCAST_VERSION节点

    TiXmlHandle docHandle( &doc );
        TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").ToElement();

        TiXmlNode * node =  Broadcast_ver->FirstChild("BROADCAST_VERSION");

        Broadcast_ver->RemoveChild(node);

    也可以删除整个SOURCE_1节点:

    TiXmlHandle docHandle( &doc );
        TiXmlElement* Broadcast = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChild("Broadcast").ToElement();

        TiXmlNode * node =  Broadcast->FirstChild("SOURCE_1");

        Broadcast->RemoveChild(node);

    删除BROADCAST_VERSION的info属性:

    TiXmlHandle docHandle( &doc );
        TiXmlElement* Broadcast_ver = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChildElement("BROADCAST_VERSION").ToElement();

        Broadcast_ver->RemoveAttribute("info"); //删除info

    可以借助NextSiblingElement()方法实现递归删除.

    三,添加节点,属性值

    例如在SOURCE_3下添加BROADCAST_PID节点:

    TiXmlHandle docHandle( &doc );
        TiXmlElement* Broadcast = docHandle.FirstChild("ZXML").FirstChild("ZAPP").FirstChild("VBS_RUNTIME_PARAMS").FirstChild("Broadcast").ToElement();
        TiXmlElement* Broadcast_Pid = new TiXmlElement("BROADCAST_PID");
        TiXmlText *text =new TiXmlText("7215");
        Broadcast_Pid->SetAttribute("info","the pid");
        Broadcast_Pid->LinkEndChild(text);
        Broadcast->LinkEndChild(Broadcast_Pid);

    将在SOURCE_3后添加新的节点:

    <BROADCAST_PID info="the pid">7215</BROADCAST_PID>

    四,最后说一下中文乱码的问题

    乱码是由于GB2312与UTF8之间转换不当造成的,tinyxml在处理UTF8本身没有问题,当你打开一个UTF8的文档,可以在加载的时候指定UTF8的方式,或者文档声明处指明的编码格式,tinyxml会按照相应的编码格式加载,但很多时候当我们输出或写入中文字段时会出现乱码,无论在内存,还是打印出来的内容.这是因为我们的软件通常是GB2312编码,而读取或写入的内容是UTF8,自然就会出错.可以借助网上的两个函数来实现转换(原作者不详):

        void ConvertUtf8ToGBK(CString& strUtf8) 
        {
            int len=MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, NULL,0);
            unsigned short * wszGBK = new unsigned short[len+1];
            memset(wszGBK, 0, len * 2 + 2);
            MultiByteToWideChar(CP_UTF8, 0, (LPCTSTR)strUtf8, -1, wszGBK, len);

            len = WideCharToMultiByte(CP_ACP, 0, wszGBK, -1, NULL, 0, NULL, NULL);
            char *szGBK=new char[len + 1];
            memset(szGBK, 0, len + 1);
            WideCharToMultiByte (CP_ACP, 0, wszGBK, -1, szGBK, len, NULL,NULL);

            strUtf8 = szGBK;
            delete[] szGBK;
            delete[] wszGBK;
        }



        void ConvertGBKToUtf8(CString& strGBK)
        {
            int len=MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, NULL,0);
            unsigned short * wszUtf8 = new unsigned short[len+1];
            memset(wszUtf8, 0, len * 2 + 2);
            MultiByteToWideChar(CP_ACP, 0, (LPCTSTR)strGBK, -1, wszUtf8, len);

            len = WideCharToMultiByte(CP_UTF8, 0, wszUtf8, -1, NULL, 0, NULL, NULL);
            char *szUtf8=new char[len + 1];
            memset(szUtf8, 0, len + 1);
            WideCharToMultiByte (CP_UTF8, 0, wszUtf8, -1, szUtf8, len, NULL,NULL);

            strGBK = szUtf8;
            delete[] szUtf8;
            delete[] wszUtf8;
        }

  • 相关阅读:
    【C语言】学习笔记9——结构struct(2)
    WPF dev 获取gridControl筛选后的数据
    WPF DEV dxc:ChartControl 柱状图
    WPF 重写ListBox(透明效果)
    WPF 后台重写 DataTemplate
    WPF 去掉Drag a column header here to group by that column
    c# 对DataTable进行分组group by
    c# ref与out用法
    WPF canvas设置旋转角度和偏移位置
    WPF 流加载
  • 原文地址:https://www.cnblogs.com/nktblog/p/4027092.html
Copyright © 2020-2023  润新知