Linux之获取配置文件行字段
2016-04-11 10:45
477 查看
工作中少不了要提取规则文件(如:日志文件/配置文件)行或行里字段信息,前一段时间在研究某应用程序源码时,发现其对
配置文件信息的提取方法很巧妙,后来工作上需要与同事之间交互信息,刚好通过日志来实现,日志的格式也是本人制定,提到
的这个方法也就用上了;
就拿最简单的配置文件来说,至少包含两部分信息:一是有用的注释,二是参数和参数设定值;
注释一般以#开头,整行都是解释信息,在读取时若是行首字节是#,则该行信息直接忽略,读取下一行;
参数和参数值的行一般参数字段定格书写,两者之间通过空格隔断;
如下示例:
# This is a test file.
cnt 1000
现在需要提取文件中cnt的设定值,方法很简单,只需要打开该文件,然后按行读取文件信息,先判断行首字节是否#,若是#则继续
读取下一行文件内容,若不是则比较该行的第一个字段是否为cnt,若是则提取该行的第二个字段就是cnt参数的设定值;
上面提到的是最基本的应用,实际任务中,阅读过源代码的都了解配置文件,每个配置文件有许多参数,这些参数的设定值开发者可以
自行设定,那么问题来了,这些设定的参数值,软件在启动时是怎样提取到的呢?方法就是上面的延伸了,将读取到的参数信息转换成链表
形式进行存储,每个节点包含成员有:参数名(char name[255]),参数值(int parameter),指针struct xxx *next,这样配置文件里面的参数
就存储到了内存的链表中,然后就可以通过提取链表中的值对内部参数赋值;到这一步应该已经没什么大问题了,但是结合最近了解的新知识
要是配置文件参数数量很多,形成的链表就会很长,查询起来将会消耗大量性能,对于这个问题,如果了解哈希链表的同学就会想到办法,这里
不多说,等研究透彻了在另外一篇博文中会提到,网上也有大量资源可查,将了这么多,看下上述中最简单方法的实现:
#include<stdio.h>
#include<string.h>
#include<fcntl.h>
#include<unistd.h>
#include<sys/types.h>
#define MAX_LEN 1024
/*Get the specified field in the string*/
char *GetNField(const char *src,char *des,int n){
int counter = 1;
const char *tmpsrc = src;
char *tmpdes = des;
if(n < 1){
//printf("Error:the number of specified field is too small\n");
return NULL;
}
/*过滤空格*/
while(*tmpsrc == ' '){
tmpsrc++;
}
/*过滤空行*/
if(*tmpsrc == '\r' || *tmpsrc == '\n' || *tmpsrc == '\0'){
return NULL;
}
/*过滤前n-1个字段*/
while(counter < n){
/*空格为字段间隔符号,\r或\n或\0为行结束符号 */
while(*tmpsrc != ' ' && (*tmpsrc != '\r' || *tmpsrc != '\n' || *tmpsrc != '\0')){
tmpsrc++;
}
/*检测行是否结束,来判断是否有指定的字段存在*/
if(*tmpsrc == '\r' || *tmpsrc == '\n' || *tmpsrc == '\0'){
//printf("Error:the number of specified field is too big\n");
return NULL;
}
/*过滤空格*/
while(*tmpsrc == ' '){
tmpsrc++;
}
counter++;
}
/*获取目标字段*/
while(*tmpsrc != ' ' && *tmpsrc != '\r' && *tmpsrc != '\n' && *tmpsrc != '\0'){
*tmpdes++ = *tmpsrc++;
}
*tmpdes = '\0';
return des;
}
/*The 1th field is the parameter name*/
char *GetParameterName(const char *linebuf,char *parameterbuf){
return GetNField(linebuf,parameterbuf,1);
}
/*The 2th field is the parameter value*/
char *GetParameterValue(const char *linebuf,char *valuebuf){
return GetNField(linebuf,valuebuf,2);
}
int main(int argc,char **argv){
char linebuf[MAX_LEN] = {0};
char parameterbuf[50] = {0};
char valuebuf[50] = {0};
int ret = 0;
FILE* fp;
/*Usage*/
if(argc != 2){
printf("Error!Usage: %s [filename]\n",argv[0]);
return -1;
}
/*File exists or not*/
ret = access(argv[1],F_OK);
if(-1 == ret){
printf("%s is not found",argv[1]);
return -1;
}
fp = fopen(argv[1],"r");
if(NULL == fp){
printf("Error:fopen %s fail!\n",argv[1]);
return -1;
}
/*按行读取文件内容,并进行检测*/
while(fgets(linebuf,MAX_LEN,fp)){
if('#' == linebuf[0]){
continue;
}
if(GetParameterName(linebuf,parameterbuf)){
GetParameterValue(linebuf,valuebuf);
}else{
continue;
}
/*这个位置可以添加转换成链表的功能,将parameter和value作为一个结构体的成员,添加到单链表中*/
/*打印获取的信息,其中值可以用atoi转换为整型数据,可以根据实
4000
际情况来定*/
printf("%s = %s\n",parameterbuf,valuebuf);
}
fclose(fp);
return 0;
}
测试文件test.txt:
#This is a test file.
#cnt = 1000
cnt 1000
#value = 123
vaule 123
#default = 666
default 666
#End of the file.
测试结果:
配置文件信息的提取方法很巧妙,后来工作上需要与同事之间交互信息,刚好通过日志来实现,日志的格式也是本人制定,提到
的这个方法也就用上了;
就拿最简单的配置文件来说,至少包含两部分信息:一是有用的注释,二是参数和参数设定值;
注释一般以#开头,整行都是解释信息,在读取时若是行首字节是#,则该行信息直接忽略,读取下一行;
参数和参数值的行一般参数字段定格书写,两者之间通过空格隔断;
如下示例:
# This is a test file.
cnt 1000
现在需要提取文件中cnt的设定值,方法很简单,只需要打开该文件,然后按行读取文件信息,先判断行首字节是否#,若是#则继续
读取下一行文件内容,若不是则比较该行的第一个字段是否为cnt,若是则提取该行的第二个字段就是cnt参数的设定值;
上面提到的是最基本的应用,实际任务中,阅读过源代码的都了解配置文件,每个配置文件有许多参数,这些参数的设定值开发者可以
自行设定,那么问题来了,这些设定的参数值,软件在启动时是怎样提取到的呢?方法就是上面的延伸了,将读取到的参数信息转换成链表
形式进行存储,每个节点包含成员有:参数名(char name[255]),参数值(int parameter),指针struct xxx *next,这样配置文件里面的参数
就存储到了内存的链表中,然后就可以通过提取链表中的值对内部参数赋值;到这一步应该已经没什么大问题了,但是结合最近了解的新知识
要是配置文件参数数量很多,形成的链表就会很长,查询起来将会消耗大量性能,对于这个问题,如果了解哈希链表的同学就会想到办法,这里
不多说,等研究透彻了在另外一篇博文中会提到,网上也有大量资源可查,将了这么多,看下上述中最简单方法的实现:
#include<stdio.h>
#include<string.h>
#include<fcntl.h>
#include<unistd.h>
#include<sys/types.h>
#define MAX_LEN 1024
/*Get the specified field in the string*/
char *GetNField(const char *src,char *des,int n){
int counter = 1;
const char *tmpsrc = src;
char *tmpdes = des;
if(n < 1){
//printf("Error:the number of specified field is too small\n");
return NULL;
}
/*过滤空格*/
while(*tmpsrc == ' '){
tmpsrc++;
}
/*过滤空行*/
if(*tmpsrc == '\r' || *tmpsrc == '\n' || *tmpsrc == '\0'){
return NULL;
}
/*过滤前n-1个字段*/
while(counter < n){
/*空格为字段间隔符号,\r或\n或\0为行结束符号 */
while(*tmpsrc != ' ' && (*tmpsrc != '\r' || *tmpsrc != '\n' || *tmpsrc != '\0')){
tmpsrc++;
}
/*检测行是否结束,来判断是否有指定的字段存在*/
if(*tmpsrc == '\r' || *tmpsrc == '\n' || *tmpsrc == '\0'){
//printf("Error:the number of specified field is too big\n");
return NULL;
}
/*过滤空格*/
while(*tmpsrc == ' '){
tmpsrc++;
}
counter++;
}
/*获取目标字段*/
while(*tmpsrc != ' ' && *tmpsrc != '\r' && *tmpsrc != '\n' && *tmpsrc != '\0'){
*tmpdes++ = *tmpsrc++;
}
*tmpdes = '\0';
return des;
}
/*The 1th field is the parameter name*/
char *GetParameterName(const char *linebuf,char *parameterbuf){
return GetNField(linebuf,parameterbuf,1);
}
/*The 2th field is the parameter value*/
char *GetParameterValue(const char *linebuf,char *valuebuf){
return GetNField(linebuf,valuebuf,2);
}
int main(int argc,char **argv){
char linebuf[MAX_LEN] = {0};
char parameterbuf[50] = {0};
char valuebuf[50] = {0};
int ret = 0;
FILE* fp;
/*Usage*/
if(argc != 2){
printf("Error!Usage: %s [filename]\n",argv[0]);
return -1;
}
/*File exists or not*/
ret = access(argv[1],F_OK);
if(-1 == ret){
printf("%s is not found",argv[1]);
return -1;
}
fp = fopen(argv[1],"r");
if(NULL == fp){
printf("Error:fopen %s fail!\n",argv[1]);
return -1;
}
/*按行读取文件内容,并进行检测*/
while(fgets(linebuf,MAX_LEN,fp)){
if('#' == linebuf[0]){
continue;
}
if(GetParameterName(linebuf,parameterbuf)){
GetParameterValue(linebuf,valuebuf);
}else{
continue;
}
/*这个位置可以添加转换成链表的功能,将parameter和value作为一个结构体的成员,添加到单链表中*/
/*打印获取的信息,其中值可以用atoi转换为整型数据,可以根据实
4000
际情况来定*/
printf("%s = %s\n",parameterbuf,valuebuf);
}
fclose(fp);
return 0;
}
测试文件test.txt:
#This is a test file.
#cnt = 1000
cnt 1000
#value = 123
vaule 123
#default = 666
default 666
#End of the file.
测试结果:
相关文章推荐
- Linux socket 初步
- Linux Kernel 4.0 RC5 发布!
- linux lsof详解
- linux 文件权限
- Linux 执行数学运算
- 10 篇对初学者和专家都有用的 Linux 命令教程
- Linux 与 Windows 对UNICODE 的处理方式
- Ubuntu12.04下QQ完美走起啊!走起啊!有木有啊!
- 解決Linux下Android开发真机调试设备不被识别问题
- 运维入门
- 运维提升
- Linux 自检和 SystemTap
- Ubuntu Linux使用体验
- c语言实现hashmap(转载)
- Linux 信号signal处理机制
- linux下mysql添加用户
- Scientific Linux 5.5 图形安装教程
- Linux 下无损图片压缩小工具介绍