• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            大龍的博客

            常用鏈接

            統(tǒng)計(jì)

            最新評(píng)論

            用STL快速編寫ini配置文件識(shí)別類

            作者:Winter

            ini文件是技術(shù)人員經(jīng)常用到的一種系統(tǒng)配置方法,如何讀取和快速識(shí)別ini文件中的內(nèi)容實(shí)現(xiàn)起來(lái)比較繁瑣。STL強(qiáng)大的功能在于能快速的實(shí)現(xiàn)排序、查找、 識(shí)別等功能。本文通過(guò)STL中的map,string,vector,ifstream等,來(lái)快速實(shí)現(xiàn)ini文件的識(shí)別類class IniFile?。IniFile可以實(shí)現(xiàn)常見查找功能,并提供完整的源碼。

            1 設(shè)計(jì)需求:

            ini文件的格式一般如下:

            [section1]
            key1=value1
            key2=value2
            ......
            
            [section2]
            key1=value1
            key2=value2    #注釋
            ......
            實(shí)際的例子是:
            
            #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值,注釋 #或者//開頭,標(biāo)志字符"[" "]" "="。查找項(xiàng)的對(duì)應(yīng)關(guān)系為sectiong-key和value對(duì)應(yīng)。需要得到是value。class IniFile?要實(shí)現(xiàn)的是兩個(gè)函數(shù):讀入ini文件,讀取sect-key對(duì)應(yīng)的value值。即實(shí)現(xiàn)下面的接口:

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

            2 設(shè)計(jì)實(shí)現(xiàn):

            用ifstream按行讀入ini文件的內(nèi)容

            識(shí)別每一行的字符串,分析出sectiong,key,value,和注釋。

            用map<string, string, less >來(lái)記錄所有的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是用來(lái)真正實(shí)現(xiàn)初始化ini內(nèi)容的函數(shù)。先用ifstream fin打開一個(gè)文件,然后用is_open判斷文件是否正常打開。順序讀取文件的時(shí)候用eof()判斷是否到文件尾。getline是一個(gè)字符處理函數(shù):直接從fin中讀取一行。然后用while循環(huán)過(guò)濾一行末尾的空格等字符。最后保存到一個(gè)vector中,完成讀入文本工作。其中比較值得關(guān)注的是以下為體,你知道為什么這么做么?

            • 用ifstream和getline來(lái)讀入而不是用fopen和fread。
            • 用is_open判斷是否打開,而不是直接讀取。
            • 用vector的push_pack而不是insert。
            • 用empty判斷是否為空,而不是用size()==0。

            下一步用for_each函數(shù)來(lái)完成字符串的內(nèi)容提取工作。聲明一個(gè)結(jié)構(gòu),實(shí)現(xiàn)對(duì)操作符()的重載。代碼如下:

            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系列,功能確實(shí)十分強(qiáng)大。所有在string中沒有找到都會(huì)返回一個(gè)變量string::npos。

            函數(shù)先找sectiong,然后分離key值和value值。符合要求的,把section和key值通過(guò)中間加上MIDDLESTRING組成一個(gè)新的string,插入map中。這里值得注意的是:

            * for_each的使用,結(jié)構(gòu)可以傳遞參數(shù)。 * string的查找函數(shù)及返回值 * string的鏈接和合并函數(shù)。 * map的下標(biāo)操作符的使用。

            3 具體使用

            把所有代碼放在一個(gè)頭文件中,以后別人使用的時(shí)候,只需要包含頭文件就可以了,點(diǎn)擊查看inifile.h文件。在使用的過(guò)程中,注意判斷返回值。使用代碼如下:

            #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配置文件識(shí)別類

            posted on 2006-10-09 17:36 大龍 閱讀(364) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            日韩十八禁一区二区久久| 亚洲综合熟女久久久30p| 久久久综合九色合综国产| 国产午夜久久影院| 久久久久久噜噜精品免费直播| 久久久久久久91精品免费观看| 久久精品无码一区二区WWW| 看久久久久久a级毛片| 久久天堂电影网| 亚洲人成精品久久久久| 7国产欧美日韩综合天堂中文久久久久| 久久一本综合| 久久久久久亚洲AV无码专区| 国产精品亚洲美女久久久| 欧美成人免费观看久久| 精品免费tv久久久久久久| 久久久高清免费视频| 久久精品国产免费一区| AV无码久久久久不卡蜜桃| 久久人妻少妇嫩草AV蜜桃| 久久国产热精品波多野结衣AV| 四虎久久影院| 久久影院久久香蕉国产线看观看| 久久久无码精品亚洲日韩按摩 | 少妇人妻88久久中文字幕| 93精91精品国产综合久久香蕉| 看久久久久久a级毛片| 久久久久久久波多野结衣高潮| 色播久久人人爽人人爽人人片aV| 久久国产一区二区| 久久99精品久久久久久hb无码| 亚洲欧美日韩中文久久| 无码国内精品久久综合88| 久久无码精品一区二区三区| 亚洲成人精品久久| 热久久国产精品| 久久精品国产亚洲综合色| 精品久久久久久国产| 久久国产精品国产自线拍免费 | 中文字幕久久精品| 欧美久久综合九色综合|