配置文件的格式如下:
key1 = value1
key2 = value2
中間及前后可以有多個空格
思路分析:
讀寫配置文件可以分成底層API接口和調用API的界面二個模塊,二個模塊間耦合度要盡量低,底層封裝的API要盡量好用。
不要讓人家寫文件的時候還需要自己去判斷配置項是否已經存在,這些功能底層API要做完善。
一 接口設計
既然是讀寫,提供給外部一個讀取的接口和一個寫入的接口就可以了。要讓二個接口方便好用
int writeCFG(const char *filename/*in*/, const char *key/*in*/, const char *value/*in*/); //寫入配置文件 void readCFG(const char *filename/*in*/, const char *key/*in*/, const char **value/*out*/); //讀取配置文件
二 框架搭建
ReadAndWrite.h
函數原型
void trim(char *strIn, char *strOut);//去除字符串前面和后面的空格 void getValue(char * keyAndValue, char * key, char * value); //根據key得到value int writeCFG(const char *filename/*in*/, const char *key/*in*/, const char *value/*in*/); //寫入配置文件 void readCFG(const char *filename/*in*/, const char *key/*in*/, const char **value/*out*/); //讀取配置文件
三 代碼實現
ReadAndWrite.c
函數實現
void trim(char *strIn, char *strOut){ char *start, *end, *temp;//定義去除空格后字符串的頭尾指針和遍歷指針 temp = strIn; while (*temp == ' '){ ++temp; } start = temp; //求得頭指針 temp = strIn + strlen(strIn) - 1; //得到原字符串最后一個字符的指針(不是'\0') while (*temp == ' '){ --temp; } end = temp; //求得尾指針 for(strIn = start; strIn <= end; ){ *strOut++ = *strIn++; } *strOut = '\0'; } void getValue(char * keyAndValue, char * key, char * value){ char *p = keyAndValue; p = strstr(keyAndValue, key);//找到key的位置,指針 if(p == NULL){ printf("沒有key %s\n", key); return ; } p += strlen(key);//指針后移到key后面 trim(p, value); p = strstr(value, "=");//找等號的位置 if(p == NULL){ printf("沒有找到=\n"); return; } p+= strlen("=");//指針后移到等號后面 trim(p, value);//刪除字符串前后的空格 }
int writeCFG(const char *filename/*in*/, const char *key/*in*/, const char *value/*in*/){ FILE *pf = NULL; char ftemp[flen] = {0}, fline[1024] = {0}, *fp; //文件緩存數組 long fsize = 0; int reg = 0; int exit = 0; int i = 0;
int flen = 8*1024; pf = fopen(filename, "r+"); if(pf == NULL){ pf = fopen(filename, "w+"); } //獲得文件大小 fseek(pf, 0, SEEK_END); // 將文件指針pf指向末尾 fsize = ftell(pf);//獲取文件開頭到pf的長度 if(fsize > flen){ printf("文件不能超過8k\n"); reg = -1; goto end; } fseek(pf, 0, SEEK_SET); //將文件指針指向開頭 //一行一行的讀,如果存在key則修改value存到緩存數組中 while(!feof(pf)){//未到文件結尾 fgets(fline, 1024, pf); if(strstr(fline, key) != NULL && exit == 1) strcpy(fline, ""); if(strstr(fline, key) != NULL && exit == 0){ //判斷key是否存在 exit = 1; sprintf(fline,"%s = %s\n", key, value); } printf("fline = %s\n", fline); strcat(ftemp, fline); } if(exit != 1){//如果不存在則把key value寫入到最后一行 sprintf(fline,"%s = %s\n", key, value); strcat(ftemp, fline); } if(pf != NULL){ fclose(pf); pf = fopen(filename, "w+"); fp = (char *)malloc(sizeof(char) * strlen(ftemp) + 1); strcpy(fp, ftemp); fp[strlen(fp) - 1] = EOF; fputs(fp, pf); if(fp != NULL){ free(fp); fp = NULL; } fclose(pf); } end : if(pf != NULL) fclose(pf); //重新創建一個以filename命名的文件 return reg; } void readCFG(const char *filename/*in*/, const char *key/*in*/, const char **value/*out*/){ FILE *pf = NULL; char line[1024] = {0}, vtemp[1024] = {0}; pf = fopen(filename, "r"); //以只讀方式打開 while(!feof(pf)){ fgets(line, 1024, pf); getValue(line, key, vtemp); if(strlen(vtemp) != 0) break; } if(strlen(vtemp) != 0){ *value = (char *)malloc(sizeof(char) * strlen(vtemp) + 1); strcpy(*value, vtemp); } else *value = NULL; if(pf != NULL) fclose(pf); }
測試界面代碼:
#define filename "c:/cfg.ini" void menu(){ printf("===========================\n"); printf("1 寫入配置文件\n"); printf("2 讀取配置文件\n"); printf("0 退出程序"); printf("===========================\n"); } int tWrite(){ char key[1024] = {0}, value[1024] = {0}; printf("請輸入key:"); scanf("%s", key); printf("請輸入value:"); scanf("%s", value); printf("\n您輸入的是:%s = %s\n", key, value); return writeCFG(filename/*in*/,key/*in*/,value/*in*/); } void tRead(){ char key[1024] = {0}, *value; printf("請輸入key:"); scanf("%s", key); readCFG(filename/*in*/,key/*in*/, &value/*out*/); if(value == NULL){ printf("沒有key\n"); return ; } printf("\nvalue = %s\n", value); if(value != NULL){ free(value); value = NULL; } } int main(){ int choose; while(1){ choose = 0; menu(); printf("請輸入選擇:"); scanf("%d", &choose); switch(choose){ case 1: if(tWrite() == -1) return -1; break; case 2: tRead(); break; case 0: return 0; default: return 0; } } system("pause"); return 0; }
