三叶草 发表于 2021-12-31 13:16:55

Linux 下读XML 的类详解及实现代码

这篇文章主要介绍了 Linux 下读XML 的类详解及实现代码的相关资料,需要的朋友可以参考下
Linux 下读XML 的类详解及实现代码
在Linux下写程序,常需要读一些配置文件。现有的XML工具很多,可以方便的编辑和生成XML。
但VC中用的XML解析器在Linux下不能用。只好自已写了个。用了一下,还不错。


#include <stdio.h>
#include <stdlib.h>


// ********************************************************************** //
// XML解析类(honghaier写于2008-11-19)
// ********************************************************************** //


struct SXMLAttrib
{
char mKeyName; //键名
char mValue; //键值
}
;




struct SXMLFrame
{
public:
char mFrameName; //帧名
intmAttrNum;//属性数量
SXMLAttrib* mAttrArray;//属性数组

SXMLFrame* mpSiblFrame; //兄弟结点
SXMLFrame* mpChiFrame;//子结点
SXMLFrame* mpParentFrame; //父结点
public:

SXMLFrame();
~SXMLFrame();

void Release_Depath();

SXMLFrame* GetFrame_Depth(char *szFrameName);

intGetChildNum();
SXMLFrame* GetChildFrame(int Index);
SXMLFrame* GetChildFrame(char *szFrameName);
SXMLFrame* GetSiblFrame();
SXMLFrame* GetParentFrame();
SXMLAttrib* GetAttrib(char *szKeyName);



bool ParseAttrString(char *szXMLString);
}
;
class CXMLFile
{
SXMLFrame mRoot;
SXMLFrame* mpCurrentFrame;
bool mbDepthClose; //闭合
private:
bool ParseFrameString(char *szXMLString);

public:
int pFile;

CXMLFile();
~CXMLFile();
void Close();
void Release();
bool Open( const char * pFileName);

SXMLFrame* GetRoot();
SXMLFrame* GetFrame_Depth(char *szFrameName);

}
;



//====================================================


SXMLFrame::SXMLFrame()
{
memset(mFrameName,0,sizeof(mFrameName));
mAttrNum = 0;
mAttrArray = NULL;
mpSiblFrame = NULL;
mpChiFrame = NULL;
mpParentFrame = NULL;
}
SXMLFrame::~SXMLFrame()
{
Release_Depath();
}

void SXMLFrame::Release_Depath()
{
if(mAttrNum > 0)
{
if(mAttrArray)
{
delete[] mAttrArray;
mAttrArray = NULL;
}
mAttrNum = 0;
}
if(mpChiFrame)
{
mpChiFrame->Release_Depath();
delete mpChiFrame;
mpChiFrame = NULL;
}
if(mpSiblFrame)
{
mpSiblFrame->Release_Depath();
delete mpSiblFrame;
mpSiblFrame = NULL;
}
}

SXMLFrame* SXMLFrame::GetFrame_Depth(char *szFrameName)
{
if(strcmp(mFrameName,szFrameName)==0)
{
return this;
}
if(mpChiFrame)
{
SXMLFrame* tResFrame = mpChiFrame->GetFrame_Depth(szFrameName);
if(tResFrame)return tResFrame;
}
if(mpSiblFrame)
{
SXMLFrame* tResFrame = mpSiblFrame->GetFrame_Depth(szFrameName);
if(tResFrame)return tResFrame;
}

return NULL;
}
int SXMLFrame::GetChildNum()
{
int count = 0;
for(SXMLFrame *temp = mpChiFrame ; temp != NULL ;temp = temp->mpSiblFrame)
{
count++;
}
return count;
}

SXMLFrame* SXMLFrame::GetChildFrame(int Index)
{
int count = 0;
for(SXMLFrame *temp = mpChiFrame ; temp != NULL ;temp = temp->mpSiblFrame)
{
if(count == Index)return temp;
count++;
}
return NULL;
}

SXMLFrame* SXMLFrame::GetChildFrame(char *szFrameName)
{
for(SXMLFrame *temp = mpChiFrame ; temp != NULL ;temp = temp->mpSiblFrame)
{
if(strcmp(temp->mFrameName,szFrameName)==0)
{
return temp;
}
}
return NULL;
}

SXMLFrame* SXMLFrame::GetSiblFrame()
{
return mpSiblFrame;
}

SXMLFrame* SXMLFrame::GetParentFrame()
{
return mpParentFrame;
}

SXMLAttrib* SXMLFrame::GetAttrib(char *szKeyName)
{
for(int i = 0 ; i < mAttrNum ; i++)
{
if(strcmp(mAttrArray.mKeyName,szKeyName)==0)
{
return &mAttrArray;
}
}
return NULL;
}

bool SXMLFrame::ParseAttrString(char *szXMLString)
{
SXMLAttrib AttribArray;
int len = strlen(szXMLString);
mAttrNum = 0;
int StrPos = 0;
bool HaveFrameName = false;
for(int i = 0 ;i < len ; i++)
{
if(i==(len-1))
{
if(false == HaveFrameName)
{
memcpy(mFrameName,szXMLString,len);
mFrameName='/0';
HaveFrameName = true;
}
else
{
if(( len - StrPos-1 )== 0)
{
   memset(AttribArray.mValue,0,sizeof(AttribArray.mValue));
}
else
{
   memcpy(AttribArray.mValue,szXMLString+StrPos,len-StrPos-1);
   AttribArray.mValue='/0';
}
mAttrNum++;
StrPos = 0;
}
break;
}
if(szXMLString == ' '&&szXMLString == ' ')
{
StrPos = i+1;
continue;
}
if(szXMLString == ' ')
{
if(false == HaveFrameName)
{
memcpy(mFrameName,szXMLString,i);
mFrameName='/0';
HaveFrameName = true;
StrPos = i+1;
continue;
}
else
{
if(( i - StrPos-1 )== 0)
{
   memset(AttribArray.mValue,0,sizeof(AttribArray.mValue));
}
else
{
   memcpy(AttribArray.mValue,szXMLString+StrPos,i-StrPos-1);
   AttribArray.mValue='/0';
}
mAttrNum++;
StrPos = i+1;
continue;
}
}

if(szXMLString == '=')
{
memcpy(AttribArray.mKeyName,szXMLString+StrPos,i-StrPos);
AttribArray.mKeyName='/0';
i++;//跳过一个"""
StrPos = i+1;
continue;
}

}

mAttrArray = new SXMLAttrib;
if(!mAttrArray)return false;
memcpy(mAttrArray,AttribArray,mAttrNum*sizeof(SXMLAttrib));
return true;
}

CXMLFile::CXMLFile()
{
pFile = 0;
mpCurrentFrame = NULL;
mbDepthClose = false;
}

CXMLFile::~CXMLFile()
{
Close();
}

void CXMLFile::Close()
{
if( pFile>0)
{
int error = close( pFile);
if( error!=0)
{
perror("close file failed");
}else
{
pFile=-1;
}
Release();
}
}
void CXMLFile::Release()
{
mRoot.Release_Depath();
}

bool CXMLFile::Open( const char * pFileName)
{
pFile =0;
pFile = open( pFileName,O_RDONLY);
if( pFile==-1)
{
perror(pFileName);
return false;
}

int num = 0;
char buffer;

bool bReadXMLString = false;
int XMLStringNum = 0;
char XMLString;
while(num = read(pFile,&buffer,1)>0)
{
if(buffer =='<')
{
bReadXMLString = true;
XMLStringNum = 0;
continue;
}
if(buffer == '>')
{
XMLString='/0';
if( false == ParseFrameString(XMLString))
{
printf("Read XML error: %s /n",XMLString);
return false;
}

bReadXMLString = false;

continue;
}
if(true == bReadXMLString)
{
XMLString = buffer;
}

}

mpCurrentFrame = NULL;
mbDepthClose = true;
return true;
}

SXMLFrame* CXMLFile::GetRoot()
{
return &mRoot;
}

SXMLFrame* CXMLFile::GetFrame_Depth(char *szFrameName)
{
return mRoot.GetFrame_Depth(szFrameName);
}

bool CXMLFile::ParseFrameString(char *szXMLString)
{
if(szXMLString == '?')return true;
if(szXMLString == '!')return true;

if(szXMLString == '/')
{
//如果是结束
mpCurrentFrame = mpCurrentFrame->GetParentFrame();
mbDepthClose = true;
}
else
{
mbDepthClose = false;

if( NULL == mpCurrentFrame)
{
mpCurrentFrame = &mRoot;
}

SXMLFrame* tNewFrame = new SXMLFrame;
tNewFrame->ParseAttrString(szXMLString);

if(false == mbDepthClose)
{
tNewFrame->mpParentFrame = mpCurrentFrame;
if( NULL == mpCurrentFrame->mpChiFrame)
{
mpCurrentFrame->mpChiFrame = tNewFrame;
}
else
{
for(SXMLFrame *temp = mpCurrentFrame->mpChiFrame ; temp != NULL ;temp = temp->mpSiblFrame)
{
   if( NULL == temp->mpSiblFrame)
   {
   temp->mpSiblFrame = tNewFrame;
   break;
   }
}
}
mpCurrentFrame = tNewFrame;
}
else
{
tNewFrame->mpParentFrame = mpCurrentFrame->GetParentFrame();
mpCurrentFrame->mpSiblFrame = tNewFrame;

mpCurrentFrame = tNewFrame;
}

}

return true;
}
用XML工具做了一个简单的XML文件。


<?xml version="1.0" encoding="UTF-8"?>
<!-- edited with XML Explorer v2.0 by Mergesoft (www.mergesoft.com)-->
<root>
<Honghaier Name="红孩儿" Age="26"></Honghaier>
</root>
在C++代码中
可以这样使用


CXMLFilexmlfile;

xmlfile.Open("1.xml");

SXMLFrame* mRootFrame = CXMLFile::GetRoot();

int ChildNum = mRootFrame ->GetChildNum();



for(int i = 0 ; i < ChildNum ; i++)

{

   SXMLFrame* tChileFrame = mRootFrame ->GetChildFrame (i);

    SXMLAttrib* tAttrib = tChileFrame->GetAttrib("Age");

    print("%s : %s= %s /n",mChileFrame ->mFrameName,tAttrib->mKeyName,tAttrib->mValue);

}
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

http://www.zzvips.com/article/216895.html
页: [1]
查看完整版本: Linux 下读XML 的类详解及实现代码