C语言解析Ini格式文件
引用别人的博文: http://www.open-open.com/lib/view/open1402278076447.html
可以解析 INI 格式的字符串、解析文件、保存到文件。
下面是头文件:
#ifndef INI_PARSER_H
#define INI_PARSER_H
#ifdef __cplusplus
extern "C" {
#endif
struct tag_value_list; struct ini_parser {
struct tag_value_list * keyvalues;
int (*parse_file)(struct ini_parser *, const char * file);
int (*parse_string)(struct ini_parser *, const char *text);
char * (*value)(struct ini_parser *, const char * key);
void (*set_value)(struct ini_parser *, const char * key, const char * value);
void (*remove)(struct ini_parser *, const char *key);
int (*save_to_file)(struct ini_parser *, const char * file);
}; struct ini_parser * new_ini_parser();
void delete_ini_parser(struct ini_parser *); #ifdef __cplusplus
}
#endif
#endif // INI_PARSER_H
下面是源文件:
#include "ini_parser.h"
#include <stdio.h>
#include <string.h>
#include "tag_value.h" static struct tag_value_pair * parse_line(char *line, int len)
{
struct tag_value_pair * pair = ;
int count = ;
char * p = line;
char * end = ;
char * start = line;
if(!p) return ;
while(*p == ' ') p++; /*blank line*/
if(p - line == len ||
*p == '\r' ||
*p == '\n' ||
*p == '\0') return ; /*do not support group*/
if(*p == '[') return ;
/*comments*/
if(*p == '#') return ; /* extract key */
start = p;
end = line + len;
while(*p != '=' && p!= end) p++;
if(p == end)
{
/* none '=' , invalid line */
return ;
}
end = p - ;
while(*end == ' ') end--; /* skip blank at the end */
count = end - start + ; pair = new_tag_value_pair();
pair->szTag = malloc(count + );
strncpy(pair->szTag, start, count);
pair->szTag[count] = ; /* extract value */
p++;
end = line + len; /* next pos of the last char */
while( *p == ' ' && p != end) p++;
if(p == end)
{
delete_tag_value_pair(pair);
return ;
}
start = p;
end--; /* to the last char */
if(*end == '\n') { *end = ; end--; }
if(*end == '\r') { *end = ; end--; }
count = end - start + ;
if(count > )
{
pair->szValue = malloc(count + );
strncpy(pair->szValue, start, count);
pair->szValue[count] = ;
} /* release empty key-value pair */
if(!pair->szValue)
{
delete_tag_value_pair(pair);
return ;
} return pair;
} static int _parse_file(struct ini_parser * ini, const char *file){
FILE * fp = fopen(file, "r");
if(fp)
{
struct tag_value_pair * pair = ;
char buf[] = {};
while(fgets(buf, , fp))
{
pair = parse_line(buf, strlen(buf));
if(pair)
{
ini->keyvalues->add(ini->keyvalues, pair);
}
}
fclose(fp);
return ini->keyvalues->size;
}
return -;
} static int _parse_text(struct ini_parser * ini, const char * text){
char *p = text;
char * start = ;
struct tag_value_pair * pair = ;
if(!text) return -; while()
{
start = p;
while(*p != '\n' && *p != '\0' )p++;
if(*p == '\0') break; pair = parse_line(start, p - start);
if(pair) ini->keyvalues->add(ini->keyvalues, pair); p++;
} return ini->keyvalues->size;
} static char * _value(struct ini_parser * ini, const char * key){
struct tag_value_pair * pair = ini->keyvalues->find_by_tag(ini->keyvalues, key);
if(pair) return pair->szValue;
return ;
} static void _set_value(struct ini_parser * ini, const char * key, const char *value){
struct tag_value_pair * pair = ini->keyvalues->find_by_tag(ini->keyvalues, key);
if(pair)
{
if(pair->szValue) free(pair->szValue);
pair->szValue = strdup(value);
}
else
{
ini->keyvalues->add(ini->keyvalues, make_tag_value_pair(key, value));
}
} static void _remove(struct ini_parser * ini, const char * key){
struct tag_value_pair * pair = ini->keyvalues->find_by_tag(ini->keyvalues, key);
if(pair)ini->keyvalues->remove(ini->keyvalues, pair);
} static void write_keyvalue(struct tag_value_pair * pair, FILE *fp)
{
fputs(pair->szTag, fp);
fputc('=', fp);
fputs(pair->szValue, fp);
fputc('\n', fp);
} static int _save_to_file(struct ini_parser * ini, const char * file){
if(ini->keyvalues->size > )
{
FILE * fp = fopen(file, "w");
if(fp)
{
struct tag_value_pair * pair = ini->keyvalues->head;
while(pair != ini->keyvalues->tail)
{
write_keyvalue(pair, fp);
pair = pair->next;
} if(pair)write_keyvalue(pair, fp); fclose(fp);
return ;
}
}
return -;
} struct ini_parser * new_ini_parser(){
struct ini_parser * ini = (struct ini_parser*)malloc(sizeof(struct ini_parser));
ini->keyvalues = new_tag_value_list();
ini->parse_file = _parse_file;
ini->parse_string = _parse_text;
ini->value = _value;
ini->set_value = _set_value;
ini->remove = _remove;
ini->save_to_file = _save_to_file;
return ini;
} void delete_ini_parser(struct ini_parser *ini){
if(ini)
{
delete_tag_value_list(ini->keyvalues);
free(ini);
}
}
测试代码:
#include "util/ini_parser.h"
#include "ini_test.h"
#include <stdio.h>
#include <assert.h> static char * g_szIniString = "#abc\nfirst=2\nsecond\nname=charli zhang \n"; static void ini_parser_test_string()
{
struct ini_parser * ini = new_ini_parser();
int size = ini->parse_string(ini, g_szIniString); assert( size > );
assert( ini->value(ini, "second") == );
assert( ini->value(ini, "abc") == );
assert( ini->value(ini, "name") != NULL );
assert( ini->value(ini, "first") != NULL); printf("ini string: %s\n", g_szIniString);
printf("key-value pairs count = %d\n", size);
printf("key \'name\'', value = %s\n", ini->value(ini, "name"));
printf("key \'first\'', value = %s\n", ini->value(ini, "first")); ini->set_value(ini, "baidu", "hahaha");
ini->save_to_file(ini, "write.conf"); ini->remove(ini, "first");
ini->save_to_file(ini, "write2.conf"); delete_ini_parser(ini);
} static void ini_parser_test_file()
{
struct ini_parser * ini = new_ini_parser();
int size = ini->parse_file(ini, "test.conf"); assert( size > );
assert( ini->value(ini, "second") == );
assert( ini->value(ini, "abc") == );
assert( ini->value(ini, "name") != NULL );
assert( ini->value(ini, "first") != NULL); printf("ini string: %s\n", g_szIniString);
printf("key-value pairs count = %d\n", size);
printf("key \'name\'', value = %s\n", ini->value(ini, "name"));
printf("key \'first\'', value = %s\n", ini->value(ini, "first"));
printf("key \'baidu\'', value = %s\n", ini->value(ini, "baidu")); delete_ini_parser(ini);
} void ini_parser_test()
{
ini_parser_test_string();
ini_parser_test_file();
}
C语言解析Ini格式文件的更多相关文章
- shell解析ini格式文件
功能 本脚本实现了ini文件中的查询修改指定value 百度云连接地址 链接:https://pan.baidu.com/s/12_T5yST7Y3L1H4_MkVEcvA 密码:fo5p 解压后先看 ...
- C语言解析WAV音频文件
C语言解析WAV音频文件 代码地址: Github : https://github.com/CasterWx/c-wave-master 目录 前言 了解WAV音频文件 什么是二进制文件 WAV的二 ...
- SHELL读取 ini 格式文件做配置文件
ini文件格式一般都是由节.键.值三部分组成 格式: [第一节 ] 第一个键 = 值 第二个键 = 第二个值 [第二节 ] 第一个键 = val1,val2,val3 例子: [COM] KINGGO ...
- 利用 nodejs 解析 m3u8 格式文件,并下 ts 合并为 mp4
利用 nodejs 解析 m3u8 格式文件,并下 ts 合并为 mp4 以前看视频的时候,直接找到 video标签,查看视频地址,然后下载下来.. 后来发现,好多 video 标签打开元素审查,如下 ...
- WP8解析XML格式文件
DOTA2 WebAPI请求返回的格式有两种,一种是XML,一种是JSON,默认是返回JSON格式,如果要返回XML格式的话,需要在加上format=xml. 这里举一个简单的解析XML格式的例子(更 ...
- dom4解析xml格式文件实例
以下给4种常见的xml文件的解析方式的分析对比: DOM DOM4J JDOM SAX 解析XML文件的几种方式和区别答: Dom解析 在内存中创建一个DOM树,该结构通常需要加载整个文档然后才能做工 ...
- dom4j解析xml格式文件实例
以下给4种常见的xml文件的解析方式的分析对比: DOM DOM4J JDOM SAX Dom解析 在内存中创建一个DOM树,该结构通常需要加载整个文档然后才能做工作.由于它是基于信息层次 ...
- Linux C语言解析.bmp格式图片并显示汉字
bmp.h 文件 #ifndef __BMP_H__ #define __BMP_H__ #include <unistd.h> #include <stdio.h> #inc ...
- XmlDocument解析Soap格式文件案例:
private static string Analysis(string strResult) { var doc = new System.Xml.XmlDocument(); //加载soap文 ...
随机推荐
- ubuntu下git安装及使用
ubuntu下git安装及使用 其实,好几个月前,就已经安装好了,可是一直搁置在那儿,所以密码等一些其它细节都忘的差不多了,所以今天就重新部署了一下,并开始积极使用......... 1,git ...
- c++学习--面向对象一实验
实验内容 一 建立类cylinder,cylinder的构造函数被传递了两个double值,分别表示圆柱体的半径和高度.用类cylinder计算圆柱体的体积,并存储在一个double变量中.在类cyl ...
- Mysql 关键字及保留字
Table 10.2 Keywords and Reserved Words in MySQL 5.7 ACCESSIBLE (R) ACCOUNT[a] ACTION ADD (R) AFTER A ...
- C#打开文件对话框
OpenFileDialog ofd = new OpenFileDialog(); ofd.InitialDirectory = System.Environment.CurrentDirector ...
- 常用MySQL命令整理
备份数据库 #.备份数据库到sql文件 mysqldump --add-drop-database -h localhost -uusername ppassword dbname > dbna ...
- 移动端自动化环境搭建-RIDE的安装
A.安装依赖 RIDE 是 Robot Framework 测试数据的编辑器.它使测试用例的创建.运行.测试项目的组织可以在图形界面下完成. B.安装过程 下载地址:https://pypi.pyth ...
- eclipse 最全快捷键 分享快乐与便捷<转发的>
Ctrl+1 快速修复(最经典的快捷键,就不用多说了) Ctrl+D: 删除当前行 Ctrl+Alt+↓ 复制当前行到下一行(复制增加) Ctrl+Alt+↑ 复制当前行到上一行(复制增加) Alt ...
- VS2010添加资源文件
VS2010中资源文件管理在 双击打开Resources.resx 选择左上角资源文件类型 然后复制资源文件 粘贴到空白区域 则会自动生成资源文件代码 在项目中使用该资源文件时,通过global::P ...
- 操作系统 页面置换算法LRU和FIFO
LRU(Least Recently Used)最少使用页面置换算法,顾名思义,就是替换掉最少使用的页面. FIFO(first in first out,先进先出)页面置换算法,这是的最早出现的置换 ...
- php阻止网页被用户频繁刷新
一般情况下,用户浏览网页的速度都是几秒十几秒甚至更长时间刷新一页,但有时候又会遇到网页被恶意快速刷新,从而导致正常用户浏览速度缓慢,如何来解决这个问题呢?可以使用如下代码来实现每ip页面访问数量限制: ...