跳至主要内容

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

作者:Winter

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

1 设计需求:

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-key和value对应。需要得到是value。class 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);   }; 

2 设计实现:

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

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

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

重新定义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中,完成读入文本工作。其中比较值得关注的是以下为体,你知道为什么这么做么?

  • 用ifstream和getline来读入而不是用fopen和fread。
  • 用is_open判断是否打开,而不是直接读取。
  • 用vector的push_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 ;     } }; 
这里大量使用了字符串的查找和字串功能。string的find_last_of系列和find系列,功能确实十分强大。所有在string中没有找到都会返回一个变量string::npos。

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

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

3 具体使用

把所有代码放在一个头文件中,以后别人使用的时候,只需要包含头文件就可以了,点击查看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; }      


  • Set MYTITLE = 用STL快速编写ini配置文件识别类

评论

此博客中的热门博文

【转】smb协议栈使用示例

/*  * * uncdownload.c  * *  * * Utility for downloading files from SMB shares using libsmbclient  * *  * * Copyright(C) 2006 Sophos Plc, Oxford, England.  * *  * * This program is free software; you can redistribute it and/or modify it under the terms of the  * * GNU General Public License Version 2 as published by the Free Software Foundation.  * *  * * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without  * * even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  * * See the GNU General Public License for more details.  * *  * * You should have received a copy of the GNU General Public License along with this program; if not,  * * write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA  * *  * */ # include < sys / types . h > # include < sys / time . h > # include ...

【转】Ether Types

Ether Types (last updated 2008-09-09) NOTE: Please see [RFC5342] for current information and registration procedures. This registry will be revised soon and will be replaced with up-to-date information. Many of the networks of all classes are Ethernets (10Mb) or Experimental Ethernets (3Mb). These systems use a message "type" field in much the same way the ARPANET uses the "link" field. If you need an Ether Type, contact: IEEE Registration Authority IEEE Standards Department 445 Hoes Lane Piscataway, NJ 08854 Phone +1 732 562 3813 Fax: +1 732 562 1571 Email: <ieee-registration-authority& ieee.org > http://standards.ieee.org/regauth/index.html The following list of EtherTypes is contributed unverified information from various sources. Another list of EtherTypes is maintained by Michael A. Patton and is accessible at: <URL: http://www.cavebear.com/CaveBear/Ethernet/ > <URL: ftp://ftp.cavebear.com/pub/Ethernet-codes > Assign...

【转】udp编程实例

UDP通讯实例 2008-04-29 15:30:05 / 个人分类: linux C编程 UDP协议的几点好处: 1.UDP不要求保持一个连接; 2.UDP没有因接收方认可收到数据包(或者当数据包没有正确抵达而自动重传)而带来的开销; 3.设计UDP的目的是用于短应用和控制信息; 4.在一个数据包接一个数据包的基础上,UDP要求的 网络 带宽比TCP更小。 UDP的几个缺点: 1.程序员必须创建代码检测传输错误并进行重传(如果应用程序要求这样做); 2.程序员必须把大数据包分片。 code: <1> /* * sender.c--UDP protocol example */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <netdb.h> int port = 6789; int main() {     int socket_descrīptor;     int iter = 0;     char buf[80];     struct sockaddr_in address;     /* Initialize socket address structure for Interner Protocols */     bzero(&address, sizeof(address)); // empty data structure     address.sin_family = AF_INET;     address.sin_addr.s_addr = inet_addr("127.0.0.1");     address...