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

简介: ini文件是技术人员经常用到的一种系统配置方法,如何读取和快速识别ini文件中的内容实现起来比较繁琐。STL强大的功能在于能快速的实现排序、查找、 识别等功能。本文通过STL中的map,string,vector,ifstream等,来快速实现ini文件的识别类class IniFile?。

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

1 设计需求:

ini文件的格式一般如下:

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

[section2]
key1=value1
key2=value2    #注释
......
实际的例子是:
None.gif#ini  for path
None.gif[path]
None.gifdictfile = /home/tmp/dict.dat
None.gifinputfile= /home/tmp/input.txt
None.gifoutputfile= /home/tmp/output.txt
None.gif
None.gif#ini  for exe
None.gif[exe]
None.gifuser= winter        // user name
None.gif
passwd= 1234567    #pass word
None.gifdatabase= mydatabase
None.gif

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

 
ExpandedBlockStart.gif class IniFile {
InBlock.gifpublic:
InBlock.gif    IniFile();
InBlock.gif    //打开ini文件
InBlock.gif
    bool open(const char* pinipath);
InBlock.gif    //读取value值
InBlock.gif
    const char* read(const char* psect, const char*pkey);
ExpandedBlockEnd.gif  }
;
None.gif

2 设计实现:

用ifstream按行读入ini文件的内容

识别每一行的字符串,分析出sectiong,key,value,和注释。

用map >来记录所有的sectiong-key和value。,>

重新定义class IniFile?

 
None.giftypedef map< stringstring, less< string> > strMap;
None.giftypedef strMap::iterator strMapIt;
None.gif
None.gif const  char* const MIDDLESTRING = "_____***_______";
None.gif class IniFile
ExpandedBlockStart.gif {
InBlock.gifpublic:
ExpandedSubBlockStart.gif    IniFile( ){};
ExpandedSubBlockStart.gif    ~IniFile( ){};
InBlock.gif    bool open(const char* pinipath)
ExpandedSubBlockStart.gif    {
InBlock.gif        return do_open(pinipath);
ExpandedSubBlockEnd.gif    }

InBlock.gif    string read(const char*psect, const char*pkey)
ExpandedSubBlockStart.gif    {
InBlock.gif        string mapkey = psect;
InBlock.gif        mapkey += MIDDLESTRING;
InBlock.gif        mapkey += pkey;
InBlock.gif        strMapIt it = c_inimap.find(mapkey);
InBlock.gif        if(it == c_inimap.end())
InBlock.gif            return "";
InBlock.gif        else
InBlock.gif            return it->second;
ExpandedSubBlockEnd.gif    }

InBlock.gifprotected:
InBlock.gif    bool do_open(const char* pinipath)
ExpandedSubBlockStart.gif    {
InBlock.gif        ifstream fin(pinipath);
InBlock.gif        if(!fin.is_open())
InBlock.gif            return false;
InBlock.gif        vector<string> strvect;
InBlock.gif        while(!fin.eof())
ExpandedSubBlockStart.gif        {
InBlock.gif            string inbuf;
InBlock.gif            getline(fin, inbuf,'\n');
InBlock.gif            strvect.push_back(inbuf);
ExpandedSubBlockEnd.gif        }

InBlock.gif        if(strvect.empty())
InBlock.gif            return false;
InBlock.gif        for_each(strvect.begin(), strvect.end(), analyzeini(c_inimap));
InBlock.gif        return !c_inimap.empty();
ExpandedSubBlockEnd.gif    }

InBlock.gif    strMap c_inimap;
ExpandedBlockEnd.gif}
;
None.gif

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

  • 用ifstream和getline来读入而不是用fopen和fread。
  • 用is_open判断是否打开,而不是直接读取。
  • 用vector的push_pack而不是insert。
  • 用empty判断是否为空,而不是用size()==0。

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

 
ExpandedBlockStart.giftruct analyzeini {
InBlock.gif    string strsect;
InBlock.gif    strMap *pmap;
ExpandedSubBlockStart.gif    analyzeini(strMap & strmap):pmap(&strmap){}
InBlock.gif    void operator()( const string & strini)
ExpandedSubBlockStart.gif    {
InBlock.gif        int first =strini.find('[');
InBlock.gif        int last = strini.rfind(']');
InBlock.gif        if( first != string::npos && last != string::npos && first != last+1)
ExpandedSubBlockStart.gif        {
InBlock.gif            strsect = strini.substr(first+1,last-first-1);
InBlock.gif            return ;
ExpandedSubBlockEnd.gif        }

InBlock.gif        if(strsect.empty())
InBlock.gif            return ;
InBlock.gif        if((first=strini.find('='))== string::npos)
InBlock.gif            return ;
InBlock.gif        string strtmp1= strini.substr(0,first);
InBlock.gif        string strtmp2=strini.substr(first+1, string::npos);
InBlock.gif        first= strtmp1.find_first_not_of(" \t");
InBlock.gif        last = strtmp1.find_last_not_of(" \t");
InBlock.gif        if(first == string::npos || last == string::npos)
InBlock.gif            return ;
InBlock.gif        string strkey = strtmp1.substr(first, last-first+1);
InBlock.gif        first = strtmp2.find_first_not_of(" \t");
InBlock.gif        if(((last = strtmp2.find("\t#", first )) != string::npos) ||
InBlock.gif            ((last = strtmp2.find(" #", first )) != string::npos) ||
InBlock.gif            ((last = strtmp2.find("\t//", first )) != string::npos)||
InBlock.gif            ((last = strtmp2.find(" //", first )) != string::npos))
ExpandedSubBlockStart.gif        {
InBlock.gif            strtmp2 = strtmp2.substr(0, last-first);
ExpandedSubBlockEnd.gif        }

InBlock.gif        last = strtmp2.find_last_not_of(" \t");
InBlock.gif        if(first == string::npos || last == string::npos)
InBlock.gif            return ;
InBlock.gif        string value = strtmp2.substr(first, last-first+1);
InBlock.gif        string mapkey = strsect + MIDDLESTRING;
InBlock.gif        mapkey += strkey;
InBlock.gif        (*pmap)[mapkey]=value;
InBlock.gif        return ;
ExpandedSubBlockEnd.gif    }

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

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

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

3 具体使用

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

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


  • Set MYTITLE = 用STL快速编写ini配置文件识别类
目录
相关文章
|
10天前
|
PHP 开发者
深入解析PHP的命名空间与自动加载机制
【4月更文挑战第30天】 在现代PHP开发实践中,命名空间和自动加载机制是模块化和代码复用的关键。本文旨在提供一个全面的视角来理解这两个概念如何协同工作以优化项目结构。我们将探讨命名空间解决代码冲突的方式,以及自动加载机制如何智能地按需加载类,从而减少内存占用和提升性能。
|
21天前
|
PHP 开发者
深入解析PHP的命名空间和自动加载机制
【4月更文挑战第19天】在PHP的编程世界中,命名空间和自动加载机制是两个核心的概念。它们不仅帮助开发者有效地管理代码,还提高了代码的重用性和可维护性。本文将深入探讨这两个概念,并解析它们的工作原理和使用方式。
|
22天前
|
关系型数据库 MySQL PHP
深入探讨PHP中的命名空间和自动加载机制
【4月更文挑战第18天】在PHP的编程世界中,命名空间和自动加载机制是两个关键的概念,它们不仅有助于代码的组织和维护,而且对于实现高效的模块化编程至关重要。本文将详细探讨PHP命名空间的作用、如何正确使用它们以及自动加载机制的原理和实践,帮助开发者提升代码的可读性和可维护性,同时减少内存资源的消耗。
16 0
|
10月前
|
PHP
php开发实战分析(4):php调用封装函数包含文件路径自适应不同目录的解决方案($_SERVER[‘DOCUMENT_ROOT‘]与__DIR__魔术常量)
php开发实战分析(4):php调用封装函数包含文件路径自适应不同目录的解决方案($_SERVER[‘DOCUMENT_ROOT‘]与__DIR__魔术常量)
133 0
|
编译器 应用服务中间件 PHP
PHP具体如何使用编译器?具体步骤是怎样的?
PHP具体如何使用编译器?具体步骤是怎样的?
107 0
|
Go
go语言简单实现加载ini文件
go语言简单实现加载ini文件
109 0
go语言简单实现加载ini文件
|
缓存 开发框架 前端开发
grape动态PHP结构(一)——目录结构与配置文件
结构的名字grape,中文名叫葡萄,因为最近一个同事经常带葡萄到公司给我们吃,受到启发想到了这个名字。
grape动态PHP结构(一)——目录结构与配置文件
|
XML JSON API
通过配置文件避免硬编码的一个例子
通过配置文件避免硬编码的一个例子
通过配置文件避免硬编码的一个例子