下载并安装libxml2
下载地址:ftp://xmlsoft.org/libxml2/
下载最新的版本,我下载的是libxml2-2.9.1.tar.gz。下载后将文件解压到合适的位置,进入解压后的目录。
编译命令非常简单(注意:如果configure文件没有可执行权限,增加可执行权限):
./configure
make
make install
此时libxml2相关的头文件应该在/usr/local/include/libxml2目录下,libxml2相关的库文件应该在/usr/local/lib目录下。
例子程序参考:
两种方式读写xml:
https://www.cnblogs.com/catgatp/p/6505451.html
操作xml汇总:
https://www.cnblogs.com/phisy/p/3373762.html
http://blog.sina.com.cn/s/blog_6a1837e90100ns9j.html
http://blog.csdn.net/pbymw8iwm/article/details/7899156
http://blog.csdn.net/guoping16/article/details/6591144
http://blog.csdn.net/yusiguyuan/article/details/12648553
详细例子:https://www.cnblogs.com/Anker/p/3542058.html
实现获取任意一条信息和修改任意一条信息。代码有冗余。
#include <stdio.h>
#include <assert.h>
#include <libxml/xmlmemory.h>
#include <libxml/parser.h>
#include <string.h>
#define DEFAULT_XML_FILE "registry.xml"
#define AM (0)
#define FM (1)
/*
typedef struct {
uint8_t* data;
int size;
} RadioName;*/
typedef struct {
int band; // Radio band:AM or FM
int freq;
char name[128];//ChannelName name;
} RadioChannel;
//解析每一个radio,提取出band、freq、redioname
static int parse_radio(xmlDocPtr doc, xmlNodePtr cur)
{
assert(doc || cur);
xmlChar *key;
cur = cur->xmlChildrenNode;
while (cur != NULL) {
//获取band
if ((!xmlStrcmp(cur->name, (const xmlChar *)"band"))) {
key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
printf("band: %s\t", key);
xmlFree(key);
}
//获取freq
if ((!xmlStrcmp(cur->name, (const xmlChar *)"freq"))) {
key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
printf("freq: %s\t", key);
xmlFree(key);
}
//获取name
if ((!xmlStrcmp(cur->name, (const xmlChar *)"name"))) {
key = xmlNodeListGetString(doc, cur->xmlChildrenNode, 1);
printf("name: %s\n", key);
xmlFree(key);
}
cur = cur->next;
}
return 0;
}
static int parse_radio_book(const char *file_name)
{
assert(file_name);
xmlDocPtr doc; //xml整个文档的树形结构
xmlNodePtr cur; //xml节点
xmlChar *id; //radio id
xmlChar *CurBand; //radio CurBand
int intCurBand = 3;
char* radio_last=NULL;
char radio_am_last[] = "radio_am_last";
char radio_fm_last[] = "radio_fm_last";
//获取树形结构
doc = xmlParseFile(file_name);
if (doc == NULL) {
fprintf(stderr, "Failed to parse xml file:%s\n", file_name);
goto FAILED;
}
//获取根节点
cur = xmlDocGetRootElement(doc);
if (cur == NULL) {
fprintf(stderr, "Root is empty.\n");
goto FAILED;
}
if ((xmlStrcmp(cur->name, (const xmlChar *)"radio_books"))) {
fprintf(stderr, "The root is not radio_books.\n");
goto FAILED;
}
//遍历处理根节点的每一个子节点
cur = cur->xmlChildrenNode;
while (cur != NULL) {
if((!xmlStrcmp(cur->name, (const xmlChar *)"cur_band_switch"))) {
CurBand = xmlGetProp(cur, "cur_band");
intCurBand = atoi(CurBand);
printf("current CurBand :%d\n",intCurBand);
if(intCurBand==0){
radio_last = radio_am_last;
}
else if(intCurBand==1){
radio_last = radio_fm_last;
}
}
/*getband getfreq */
if((!xmlStrcmp(cur->name, (const xmlChar *)(radio_last)))) {
//id = xmlGetProp(cur, "id");
//printf("current id:%s\n",id);
id = xmlGetProp(cur, "band");
printf("current band:%s\n",id);
id = xmlGetProp(cur, "freq");
printf("current freq:%s\n",id);
id = xmlGetProp(cur, "name");
printf("current name:%s\n",id);
}
/*
if ((!xmlStrcmp(cur->name, (const xmlChar *)"radio"))) {
id = xmlGetProp(cur, "name");
printf("name:%s\n",id);
xmlSetProp(cur, (const xmlChar *)"name", (const xmlChar *)"v1");
id = xmlGetProp(cur, "name");
printf("name:%s\n",id);
parse_radio(doc, cur);
xmlSaveFile(file_name, doc);
}*/
cur = cur->next;
}
xmlFreeDoc(doc);
return 0;
FAILED:
if (doc) {
xmlFreeDoc(doc);
}
return -1;
}
int getChannelList(int band){//调试参数,实际参数是getChannelListParam_t
assert(DEFAULT_XML_FILE);
xmlDocPtr doc; //xml整个文档的树形结构
xmlNodePtr cur; //xml节点
xmlChar *id; //radio id
//获取树形结构
doc = xmlParseFile(DEFAULT_XML_FILE);
if (doc == NULL) {
fprintf(stderr, "Failed to parse xml file:%s\n", DEFAULT_XML_FILE);
goto FAILED;
}
//获取根节点
cur = xmlDocGetRootElement(doc);
if (cur == NULL) {
fprintf(stderr, "Root is empty.\n");
goto FAILED;
}
if ((xmlStrcmp(cur->name, (const xmlChar *)"radio_books"))) {
fprintf(stderr, "The root is not radio_books.\n");
goto FAILED;
}
//遍历处理根节点的每一个子节点
cur = cur->xmlChildrenNode;
while (cur != NULL) {
/*getband getfreq */
if((!xmlStrcmp(cur->name, (const xmlChar *)"radio"))) {
//printf(" ==:%d\t",atoi(xmlGetProp(cur, "band")));
if((atoi(xmlGetProp(cur, "band")))==band){
id = xmlGetProp(cur, "id");
printf(" id:%s\t",id);
id = xmlGetProp(cur, "band");
printf(" band:%s\t",id);
id = xmlGetProp(cur, "freq");
printf(" freq:%s\t",id);
id = xmlGetProp(cur, "name");
printf(" name:%s\n",id);
}
}
cur = cur->next;
}
xmlFreeDoc(doc);
return 0;
FAILED:
if (doc) {
xmlFreeDoc(doc);
}
return -1;
}
RadioChannel getLastChannel(int band,RadioChannel radioChannel){//调试参数,实际参数是getChannelListParam_t
assert(DEFAULT_XML_FILE);
xmlDocPtr doc; //xml整个文档的树形结构
xmlNodePtr cur; //xml节点
xmlChar *id; //radio id
//获取树形结构
doc = xmlParseFile(DEFAULT_XML_FILE);
if (doc == NULL) {
fprintf(stderr, "Failed to parse xml file:%s\n", DEFAULT_XML_FILE);
goto FAILED;
}
//获取根节点
cur = xmlDocGetRootElement(doc);
if (cur == NULL) {
fprintf(stderr, "Root is empty.\n");
goto FAILED;
}
if ((xmlStrcmp(cur->name, (const xmlChar *)"radio_books"))) {
fprintf(stderr, "The root is not radio_books.\n");
goto FAILED;
}
//遍历处理根节点的每一个子节点
cur = cur->xmlChildrenNode;
while (cur != NULL) {
/*getband getfreq */
if((!xmlStrcmp(cur->name, (const xmlChar *)"radio_am_last"))) {
//printf(" ==:%d\t",atoi(xmlGetProp(cur, "band")));
if((atoi(xmlGetProp(cur, "band")))==band){
id = xmlGetProp(cur, "id");
//printf(" id:%s\t",id);
id = xmlGetProp(cur, "band");
//printf(" band:%s\t",id);
radioChannel.band=atoi(id);
id = xmlGetProp(cur, "freq");
//printf(" freq:%s\t",id);
radioChannel.freq=atoi(id);
id = xmlGetProp(cur, "name");
//printf(" name:%s\n",id);
strcpy(radioChannel.name,id);
}
}
cur = cur->next;
}
xmlFreeDoc(doc);
return radioChannel;
FAILED:
if (doc) {
xmlFreeDoc(doc);
}
return radioChannel;
}
static int setInfo(const char *file_name,char* info,int band,int index,char* nameChange)
{
assert(file_name);
xmlDocPtr doc; //xml整个文档的树形结构
xmlNodePtr cur; //xml节点
xmlChar *id; //radio id
xmlChar *CurBand; //radio CurBand
int intCurBand = 3;
char* radio_last=NULL;
char radio_am_last[] = "radio_am_last";
char radio_fm_last[] = "radio_fm_last";
//获取树形结构
doc = xmlParseFile(file_name);
if (doc == NULL) {
fprintf(stderr, "Failed to parse xml file:%s\n", file_name);
goto FAILED;
}
//获取根节点
cur = xmlDocGetRootElement(doc);
if (cur == NULL) {
fprintf(stderr, "Root is empty.\n");
goto FAILED;
}
if ((xmlStrcmp(cur->name, (const xmlChar *)"radio_books"))) {
fprintf(stderr, "The root is not radio_books.\n");
goto FAILED;
}
//遍历处理根节点的每一个子节点
cur = cur->xmlChildrenNode;
while (cur != NULL) {
/**/
if ((!xmlStrcmp(cur->name, (const xmlChar *)(info)))&&(atoi(xmlGetProp(cur, "id"))==index)&&(atoi(xmlGetProp(cur, "band"))==band)) {
id = xmlGetProp(cur, "name");
printf("change before:%s\n",id);
xmlSetProp(cur, (const xmlChar *)"name", (const xmlChar *)nameChange);
id = xmlGetProp(cur, "name");
printf("change after:%s\n",id);
parse_radio(doc, cur);
xmlSaveFile(file_name, doc);
}
cur = cur->next;
}
xmlFreeDoc(doc);
return 0;
FAILED:
if (doc) {
xmlFreeDoc(doc);
}
return -1;
}
RadioChannel getInfo(const char *file_name,char* info,int band,int index,RadioChannel radioChannel)
{
assert(file_name);
xmlDocPtr doc; //xml整个文档的树形结构
xmlNodePtr cur; //xml节点
xmlChar *id; //radio id
xmlChar *CurBand; //radio CurBand
int intCurBand = 3;
char* radio_last=NULL;
char radio_am_last[] = "radio_am_last";
char radio_fm_last[] = "radio_fm_last";
//获取树形结构
doc = xmlParseFile(file_name);
if (doc == NULL) {
fprintf(stderr, "Failed to parse xml file:%s\n", file_name);
goto FAILED;
}
//获取根节点
cur = xmlDocGetRootElement(doc);
if (cur == NULL) {
fprintf(stderr, "Root is empty.\n");
goto FAILED;
}
if ((xmlStrcmp(cur->name, (const xmlChar *)"radio_books"))) {
fprintf(stderr, "The root is not radio_books.\n");
goto FAILED;
}
//遍历处理根节点的每一个子节点
cur = cur->xmlChildrenNode;
while (cur != NULL) {
/**/
if ((!xmlStrcmp(cur->name, (const xmlChar *)(info)))&&(atoi(xmlGetProp(cur, "id"))==index)&&(atoi(xmlGetProp(cur, "band"))==band)) {
id = xmlGetProp(cur, "id");
id = xmlGetProp(cur, "band");
radioChannel.band=atoi(id);
id = xmlGetProp(cur, "freq");
radioChannel.freq=atoi(id);
id = xmlGetProp(cur, "name");
strcpy(radioChannel.name,id);
}
cur = cur->next;
}
xmlFreeDoc(doc);
return radioChannel;
FAILED:
if (doc) {
xmlFreeDoc(doc);
}
return radioChannel;
}
int main(int argc, char*argv[])
{
int i;
//getChannelList(AM);
//printf("--------------\n");
//getChannelList(FM);
//getLastChannel(AM);
//getLastChannel(FM);
/**/
//RadioChannel radioChannel={0,0,'0'};
//radioChannel = getLastChannel(AM,radioChannel);
//printf("band:%d\n",radioChannel.band);
//printf("band:%d\n",radioChannel.freq);
//printf("band:%s\n",radioChannel.name);
RadioChannel radioChannel={0,0,'0'};
radioChannel=getInfo(DEFAULT_XML_FILE,"radio",0,3,radioChannel);
printf("band:%s\n",radioChannel.name);
setInfo(DEFAULT_XML_FILE,"radio",AM,3,"test");
/*
char *xml_file = DEFAULT_XML_FILE;
if (argc == 2) {
xml_file = argv[1];
}
if (parse_radio_book(xml_file) != 0) {
fprintf(stderr, "Failed to parse radio book.\n");
return -1;
}
*/
return 0;
}
<?xml version="1.0" encoding="UTF-8"?>
<radio_books>
<cur_band_switch cur_band="0"/>
<radio_am_last id="100" band="0" freq="111.88" name="v1"/>
<radio_fm_last id="100" band="1" freq="112.88" name="v1"/>
<radio id="1" band="0" freq="91.88" name="channel_1"/>
<radio id="2" band="0" freq="92.88" name="channel_2"/>
<radio id="3" band="0" freq="93.88" name="test"/>
<radio id="4" band="0" freq="94.88" name="channel_4"/>
<radio id="5" band="0" freq="91.88" name="channel_5"/>
<radio id="6" band="0" freq="92.88" name="channel_6"/>
<radio id="7" band="0" freq="93.88" name="channel_7"/>
<radio id="8" band="0" freq="94.88" name="channel_8"/>
<radio id="9" band="0" freq="91.88" name="channel_9"/>
<radio id="10" band="0" freq="92.88" name="channel_10"/>
<radio id="11" band="0" freq="93.88" name="channel_11"/>
<radio id="12" band="0" freq="94.88" name="channel_12"/>
<radio id="1" band="1" freq="91.88" name="channel_1"/>
<radio id="2" band="1" freq="92.88" name="channel_2"/>
<radio id="3" band="1" freq="93.88" name="channel_3"/>
<radio id="4" band="1" freq="94.88" name="channel_4"/>
<radio id="5" band="1" freq="91.88" name="channel_5"/>
<radio id="6" band="1" freq="92.88" name="channel_6"/>
<radio id="7" band="1" freq="93.88" name="channel_7"/>
<radio id="8" band="1" freq="94.88" name="channel_8"/>
<radio id="9" band="1" freq="91.88" name="channel_9"/>
<radio id="10" band="1" freq="92.88" name="channel_10"/>
<radio id="11" band="1" freq="93.88" name="channel_11"/>
<radio id="12" band="1" freq="94.88" name="channel_12"/>
<radio id="13" band="1" freq="92.88" name="channel_13"/>
<radio id="14" band="1" freq="93.88" name="channel_14"/>
<radio id="15" band="1" freq="94.88" name="channel_15"/>
<radio id="16" band="1" freq="92.88" name="channel_16"/>
<radio id="17" band="1" freq="93.88" name="channel_17"/>
<radio id="18" band="1" freq="94.88" name="channel_18"/>
</radio_books>