用STL快速编写ini配置文件识别类

STL快速编写ini配置文件识别类

作者: winter

ini文件是技术人员经常用到的一种系统配置方法,如何读取和快速识别ini文件中的内容实现起来比较繁琐。STL强大的功能在于能快速的实现排序、查找、 识别等功能。本文通过STL中的mapstringvector,ifstream等,来快速实现ini文件的识别类class IniFileIniFile可以实现常见查找功能,并提供完整的源码。

设计需求

ini文件的格式一般如下:

[section1]
key1=value1
key2=value2
......

[section2]
key1=value1
key2=value2    #
注释
......

实际的例子是:

#ini for path
[path]
dictfile = /home/tmp/dict.dat
inputfile= /home/tmp/input.txt
outputfile= /home/tmp/output.txt

#ini for exe
[exe]
user= winter       //user name
passwd= 1234567    #pass word
database= mydatabase

其中有五种元素:section 名,Key名,value值,注释 #或者//开头,标志字符"[" "]" "="。查找项的对应关系为sectiong-keyvalue对应。需要得到是valueclass IniFile要实现的是两个函数:读入ini文件,读取sect-key对应的value值。即实现下面的接口:

class IniFile{
public:
   
IniFile();
 
  //打开ini文件
   
bool open(const char* pinipath);
 
  //读取value
   
const char* read(const char* psect, const char*pkey);
  };

设计实现:

ifstream按行读入ini文件的内容
识别每一行的字符串,分析出sectiongkeyvalue,和注释。
map<string, string, less<string> >来记录所有的sectiongkeyvalue

重新定义class IniFile

typedef map<string, string, less<string> > strMap;
typedef strMap::iterator strMapIt;

const char*const MIDDLESTRING = "_____***_______";
class IniFile
{
public:
    IniFile( ){};
    ~IniFile( ){};
    bool open(const char* pinipath)
    {
       
return do_open(pinipath);
    }
    string read(const char*psect, const char*pkey)
    {
       
string mapkey = psect;
        mapkey += MIDDLESTRING;
        mapkey += pkey;
   
    strMapIt it = c_inimap.find(mapkey);
   
    if(it == c_inimap.end())
       
    return "";
   
    else
       
    return it->second;
    }
protected:
    bool do_open(const char* pinipath)
    {
        ifstream fin(pinipath);
        if(!fin.is_open())
            return
 false;
        vector<string> strvect;
        while(!fin.eof())
        {
            string inbuf;
            getline(fin, inbuf,'/n');
            strvect.push_back(inbuf);
        }
        if(strvect.empty())
            return false;
        for_each(strvect.begin(), strvect.end(), analyzeini(c_inimap));
        return !c_inimap.empty();
    }
    strMap c_inimap;
};

其中do_open是用来真正实现初始化ini内容的函数。先用ifstream fin打开一个文件,然后用is_open判断文件是否正常打开。顺序读取文件的时候用eof()判断是否到文件尾。getline是一个字符处理函数:直接从fin中读取一行。然后用while循环过滤一行末尾的空格等字符。最后保存到一个vector中,完成读入文本工作。其中比较值得关注的是以下为体,你知道为什么这么做么?

  • ifstreamgetline来读入而不是用fopenfread 
  • is_open判断是否打开,而不是直接读取。 
  • vectorpush_pack而不是insert 
  • empty判断是否为空,而不是用size()==0 

下一步用for_each函数来完成字符串的内容提取工作。声明一个结构,实现对操作符()的重载。代码如下:

truct analyzeini{
    string strsect;
    strMap *pmap;
    analyzeini(strMap & strmap):pmap(&strmap){}
    void operator()( const string & strini)
    {
        int first =strini.find('[');
        int last = strini.rfind(']');
        if( first != string::npos && last != string::npos && first != last+1)
        {
            strsect = strini.substr(first+1,last-first-1);
            return ;
        }
        if(strsect.empty())
            return ;
        if((first=strini.find('='))== string::npos)
            return ;
        string strtmp1= strini.substr(0,first);
        string strtmp2=strini.substr(first+1, string::npos);
        first= strtmp1.find_first_not_of(" /t");
        last = strtmp1.find_last_not_of(" /t");
        if(first == string::npos || last == string::npos)
            return ;
        string strkey = strtmp1.substr(first, last-first+1);
        first = strtmp2.find_first_not_of(" /t");
        if(((last = strtmp2.find("/t#", first )) != string::npos) ||
            ((last = strtmp2.find(" #", first )) !=
string::npos) ||
            ((last = strtmp2.find("/t//", first )) !=
string::npos)||
            ((last = strtmp2.find(" //", first )) !=
string::npos))
        {
            strtmp2 = strtmp2.substr(0, last-first);
        }
        last = strtmp2.find_last_not_of(" /t");
        if(first == string::npos || last == string::npos)
            return ;
        string value = strtmp2.substr(first, last-first+1);
        string mapkey = strsect + MIDDLESTRING;
        mapkey += strkey;
        (*pmap)[mapkey]=value;
        return ;
    }
};

这里大量使用了字符串的查找和字串功能。stringfind_last_of系列和find系列,功能确实十分强大。所有在string中没有找到都会返回一个变量string::npos

函数先找sectiong,然后分离key值和value值。符合要求的,把sectionkey值通过中间加上MIDDLESTRING组成一个新的string,插入map中。这里值得注意的是:

  • for_each的使用,结构可以传递参数。 
  • string的查找函数及返回值 
  • string的链接和合并函数。 
  • map的下标操作符的使用。 

具体使用

把所有代码放在一个头文件中,以后别人使用的时候,只需要包含头文件就可以了,点击查看inifile.h文件。在使用的过程中,注意判断返回值。使用代码如下:

#include <iostream>
#include "inifile.h"
using namespace std;
int main()
{
    IniFile ini;
   
if(!ini.open("test.ini"))
      
return -1;
   
string strvalue = ini.read("sect1","key1");
   
if(strvalue.empty())
       
return -1;
    else
        cout<<"value="<<strvalue<<endl;
   
return 0;

 

 
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值