您的位置:首页 > 运维架构 > Linux

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.

测试结果:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息