在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的一些操作。
例如:
<?
xmlversion="1.0"standalone=no> <!–Ourtodolistdata–> <ToDo> <Itempriority="1">Gotothe<bold>Toystore!</bold></Item> <Itempriority="2">Dobills</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) { //Finallydosomethinguseful.
用句柄的话就不会这么冗长了,使用TiXmlHandle类,前面的代码就会变成这样:
TiXmlHandledocHandle(
&
document); TiXmlElement
*
child2
=
docHandle.FirstChild(
"
Document
"
).FirstChild(
"
Element
"
).Child(
"
Child
"
,
1
).ToElement();
if
(child2)
{ dosomethinguseful
一、读取XML,设置节点文本 如下XML片段:
xmlversion="1.0"encoding="UTF-8"standalone="yes"
?>
<
ZXML
>
ZAPP
VBS_RUNTIME_PARAMS
BROADCAST_VERSION
info
="版本"
>
8
</
BROADCAST_VERSION
Broadcast
FileCount
="资源文件个数"
>
69
FileCount
SOURCE_1
ID
="图片编号"
>
1
ID
Version
="图片版本"
Version
Path
="图片路径"
>
/mnt/share/1.bmp
Path
FileMode
="文件处理模式"
>
0
FileMode
SOURCE_2
>
/mnt/share/2.bmp
>
2
>
.
>
要设置BROADCAST_VERSION节点的值 8为其他值,可参考如下代码(将值加1): 用ReplaceChild( TiXmlNode* replaceThis,const TiXmlNode& withThis )方法替换
TiXmlDocumentdoc("zapp.conf"); doc.LoadFile(); TiXmlHandledocHandle(
&
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); CStringnewVer; newVer.Format(
"
%d
"
,oldVer
+
1
); TiXmlTextnewText(newVer); Broadcast_ver
->
ReplaceChild(oldnode,newText); AfxMessageBox(Broadcast_ver
->
GetText());
//
输出值
doc.SaveFile();
二,删除节点,属性值
RemoveChild( TiXmlNode* removeThis )方法删除父节点的子节点, RemoveAttribute( const char * name )方法删除属性值.
例如删除BROADCAST_VERSION节点
TiXmlHandledocHandle(
&
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节点:
TiXmlHandledocHandle(
&
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属性:
TiXmlHandledocHandle(
&
doc); TiXmlElement
*
Broadcast_ver
=
docHandle.FirstChild(
"
ZXML
"
).FirstChild(
"
ZAPP
"
).FirstChild(
"
VBS_RUNTIME_PARAMS
"
).FirstChildElement(
"
BROADCAST_VERSION
"
).ToElement(); Broadcast_ver
->
RemoveAttribute(
"
info
"
);
删除info
可以借助NextSiblingElement()方法实现递归删除.
三,添加节点,242)"> 例如在SOURCE_3下添加BROADCAST_PID节点:
TiXmlHandledocHandle(
&
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
"
,
"
thepid
"
); Broadcast_Pid
->
LinkEndChild(text); Broadcast
->
LinkEndChild(Broadcast_Pid);
将在SOURCE_3后添加新的节点:
BROADCAST_PID
="thepid"
>
7215
BROADCAST_PID
四,最后说一下中文乱码的问题 乱码是由于GB2312与UTF8之间转换不当造成的,tinyxml在处理UTF8本身没有问题,当你打开一个UTF8的文档,可以在加载的时候指定UTF8的方式,或者文档声明处指明的编码格式,tinyxml会按照相应的编码格式加载,但很多时候当我们输出或写入中文字段时会出现乱码,无论在内存,还是打印出来的内容.这是因为我们的软件通常是GB2312编码,而读取或写入的内容是UTF8,自然就会出错.可以借助网上的两个函数来实现转换(原作者不详):
void
ConvertUtf8ToGBK(CString
&
strUtf8)
{ intlen=MultiByteToWideChar(CP_UTF8,0,(LPCTSTR)strUtf8,-1,NULL,0); unsignedshort*wszGBK=newunsignedshort[len+1]; memset(wszGBK,len*2+2); MultiByteToWideChar(CP_UTF8,wszGBK,len); len=WideCharToMultiByte(CP_ACP,NULL); char*szGBK=newchar[len+1]; memset(szGBK,len+1); WideCharToMultiByte(CP_ACP,szGBK,len,NULL); strUtf8=szGBK; delete[]szGBK; delete[]wszGBK; }
void
ConvertGBKToUtf8(CString
&
strGBK)
{ intlen=MultiByteToWideChar(CP_ACP,(LPCTSTR)strGBK,255)">short*wszUtf8=short[len+1]; memset(wszUtf8,len*2+2); MultiByteToWideChar(CP_ACP,wszUtf8,len); len=WideCharToMultiByte(CP_UTF8,255)">char*szUtf8=char[len+1]; memset(szUtf8,len+1); WideCharToMultiByte(CP_UTF8,szUtf8,NULL); strGBK=szUtf8; delete[]szUtf8; delete[]wszUtf8; }
当然,你也可以用MultiByteToWideChar,WideCharToMultiByte函数自己实现转换.以上是简单应用的几个举例,理解他们,相信你已经能写出满足自己需要的代码了. (编辑:李大同)
【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!
|