• <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ái)-demons
            C++/OS/disassembly/Anti-virus
            posts - 5,  comments - 5,  trackbacks - 0

            Introduction:

            In this article , you’ll learn about two points on file operation. 1. How to delete the specified line of a file. 2.How to replace the specified line of a file.

               昨天有人問我如何刪除文件中指定的一行(控制臺(tái)下),我想這個(gè)問題在GUI下和CUI下還有點(diǎn)不同,我實(shí)驗(yàn)了一下,感覺需要注意的地方還是挺多,所以我把心得整理一下,如有錯(cuò)誤,歡迎指正。

               若在GUI下,我們?cè)诟鞣N控件中編輯文本時(shí),按回車換行時(shí)會(huì)產(chǎn)生兩個(gè)字符"\r\n", 而在CUI下則取決于我們寫入時(shí)的格式,當(dāng)然如果是在CUI下,換行時(shí)我們當(dāng)然不應(yīng)該把用"\r\n"做換行符,因?yàn)槲覀兺耆梢灾挥靡粋€(gè)'\n'就可以,沒必要再浪費(fèi)多余一個(gè)字符了。當(dāng)然本文給出的刪除指定行的方法同樣適用于"\r\n"換行的字符串,但本文主要針對(duì)CUI的。相反替換指定行的方法適用是有很多限制的,下面具體討論。

            一: 刪除指定行

            思路:兩種方法,一種是把文件指針定位到要?jiǎng)h除的行首,算出該行的長度,然后用空格替換掉每一個(gè)字符。另一種是把整個(gè)文件輸入一個(gè)字符數(shù)組,然后把它當(dāng)作字符串處理,然后定位兩個(gè)位置,一個(gè)是要提換行的行首,另一個(gè)是下一行的行首,然后在把下一行到結(jié)束的所有字符平移到將要替換行的行首,也就是覆蓋法,用strcpy可輕松實(shí)現(xiàn)。

            分析:若用第一種方法,優(yōu)點(diǎn)是不用讀取整個(gè)文件(通常只需要讀到所要?jiǎng)h除行的下一行,下面代碼中有分析。)還有就是不用把所要?jiǎng)h除行的后面的全部文件內(nèi)容整體向前平移。缺點(diǎn)就是不是真正的刪除,而是把要?jiǎng)h除行的內(nèi)容用空格替換掉了,說以文件大小不會(huì)減小,當(dāng)然這也是其致命弱點(diǎn),但如果有替換某行內(nèi)容的需要,那就是它了。第二種方法是真正刪除了指定行,因?yàn)橐盐募?nèi)容全部讀入到一個(gè)字符數(shù)組,并且還要平移,所以性能上不如第一種,但它是真正的刪除了,還有就是第二種方法是把文件內(nèi)容當(dāng)作字符串來處理的,C++文件流輸出流在往文件寫字符串時(shí)是不帶結(jié)束符'\0'的,也就是如果在寫文件時(shí)沒有顯示寫'\0’,文件中是不會(huì)有'\0'的,也就是說當(dāng)我們把一個(gè)文件所有字符讀入一個(gè)字符數(shù)組時(shí),這個(gè)數(shù)組中是不會(huì)有'\0'的,說以在讀完文件內(nèi)容后,我們得在結(jié)尾加一個(gè)'\0',這樣才能用字符串處理函數(shù)來處理。好了我們先來實(shí)現(xiàn)第二種方法。

            delLine函數(shù):

            示例字符串:"aaaaa\nbbbbb\nccccc\n"

            思路有了,我們來想想怎么實(shí)現(xiàn),我們還有一個(gè)問題沒有解決,就是如何確定要替換行的首末位置,比如現(xiàn)在要?jiǎng)h除第二行,我們則必須確定我用綠色標(biāo)出的兩個(gè)位置,然后再把這兩個(gè)位置傳給strcpy,我們可以用字符查找函數(shù)strchr(字符串指針,'\n')找出這兩個(gè)位置,若找到,函數(shù)返回的是指向'\n'的指針,若沒找到,返回NULL,這里要注意一個(gè)特例,就是如果我們要?jiǎng)h除第一行時(shí),只需查找到第一個(gè)'\n'即可。好了,有了以說明,我寫了delLine函數(shù),原型是void delLine(char* pFilepath , int nLine),第一個(gè)是文件路徑,第二個(gè)是要?jiǎng)h除的行,注意,在調(diào)用該函數(shù)前一定先要關(guān)閉掉打開該文件的所有流,這是系統(tǒng)文件共享的策略決定的,因?yàn)閐elLine中要對(duì)文件進(jìn)行讀寫,如果調(diào)用該函數(shù)前,某個(gè)寫該文件的流沒關(guān)閉,這delLine函數(shù)中寫文件時(shí)會(huì)打開文件失敗,因?yàn)橄到y(tǒng)不用許同時(shí)多個(gè)用戶寫文件的。下面給出代碼:

            void delLine(char* pFilepath , int nLine)

                //Open file
                ifstream file(pFilepath);
                int  i=0;
                //Allocate read buffer. Note: can't more than 1 MB
                char buff[1024*30];
                while(!file.eof())//Read whole file
                    buff[i++]=file.get(); 
                file.close();
                buff[i]='\0';
                char *pLine=buff;
                int j=0;
                //Search newline character '\n'
                while(j++ < nLine-1 && pLine)
                {
                    pLine=strchr(pLine+1,'\n' );
                }
                if(pLine)
                {
                    char *pNextLine=strchr(pLine+1,'\n' );
                    //Cover the line we want to delete.
                    if(pNextLine){
                        if (nLine==1)
                            strcpy(pLine,pNextLine+1);
                        else
                            strcpy(pLine,pNextLine);
                    }
                }
                //we did it , write  the file handled.
                ofstream out(pFilepath);
                j=0;
                i= strlen(buff);
                while(i--)
                    out.put(buff[j++]);
                out.close();
            }
            好了,下面實(shí)現(xiàn)方案一,方案一種的核心是確定要替換行的長度,用什么方法呢? 想想,我是連續(xù)調(diào)用兩次getline函數(shù),沒調(diào)用一次getline再緊接著調(diào)用tellg得到當(dāng)前文件指針的文值,然后再用后一個(gè)位置減去前一個(gè)位置,這樣就得到了一個(gè)長度,這個(gè)長度并不是字符串的長度,來看這個(gè)字符串“aaaaa\nbbbbb\nccccc\n”,加入現(xiàn)在要替換第二行,第一次調(diào)用得到的位置是7,也就是第一個(gè)‘\n’后的一個(gè)字符位置,也就是下一行的首地址,第二次調(diào)用getline得到的位置是第一個(gè)c 的位置,也就是14,如果我們要替換的話只需保把5個(gè)b替換掉,顯然應(yīng)該是14-7-2=5,說以我們得給兩次長度差再減2. 還需要注意的是但第二次調(diào)用getline時(shí)文件指針已經(jīng)移向后一行了,所以得調(diào)用seekg調(diào)整文件指針。同樣要替換第一行時(shí)也是個(gè)特殊情況,只需調(diào)用一次getline就能確定第一行的長度。還有就是你要替換的行字符串必須大于或等于原行長度,當(dāng)大于原行長度時(shí),則超過的部分會(huì)被截?cái)啵∮跁r(shí),由于不能完全替換,應(yīng)給出提示,所以調(diào)用時(shí)傳遞的新字符串寧長勿段,我們可以在有效字符后面加許多空格,這樣就萬事大吉了。下面給出代碼:

             

            void replaceLine( char* pFilepath , int nLine,char * pReplace) 
            {
                nLine--;
                fstream file(pFilepath);
                char buff[100];
                fstream::pos_type pos, len;
                int i=0, k=0;
                //if the first line
                if (nLine==0)
                {
                    pos=file.tellg();
                    if(!file.eof())
                        file.getline(buff,100);
                    len=file.tellg()-pos;//work out the length of the line we want to delete
                    file.seekg(pos);//set the file pointer to the start of the line to be deleted
                }
                //Not first line 
                else{
                    while(i++<nLine)
                    {
                        if(!file.eof())
                            file.getline(buff,100);
                        pos=file.tellg();
                        if(!file.eof())
                            file.getline(buff,100);
                        len=file.tellg()-pos;//work out the length of the line we want to delete
                        file.seekg(pos);//set the file pointer to the start of the line to be deleted
                    }
                }

                k=len;
                
                //Note: we assume there is not '\r' before '\n', it's depend on the write format of you .
                if(strlen(pReplace)<k-2) 
                {
                    printf("Error:new line must be longer than or equal to the original line\n");
                    return ;
                }
                file.write(pReplace,k-2);
                file.close();
            }

            好了,下面我們來進(jìn)行測(cè)試:

            把第一行替換成eeeee,第二行替換成f,注意此時(shí)由于f只有一個(gè)字符,所以根據(jù)寧長勿短原則,我們應(yīng)該補(bǔ)上n個(gè)空格,在replaceLine 內(nèi)部會(huì)自動(dòng)截取原行長度個(gè)字符的,然后我們刪除第三行。

            void main(){
                //Create a file
                ofstream out("now.txt");
                out<<"aaaaa\n"<<"bbbbb\n"<<"ccccc\n"<<"ddddd\n";
                out.close();
                //Replace the first line 
                char *pReplace="eeeeeeeeeeeeeee";
                replaceLine("now.txt",1,pReplace);
                pReplace="f                              ";
                replaceLine("now.txt",2,pReplace);
                delLine("now.txt",3);
                ifstream in("now.txt");
                cout<<in.rdbuf();
            }

            輸出結(jié)果是:

            eeeee

            f

            ddddd

            該收筆了,還是那句話,轉(zhuǎn)載復(fù)制請(qǐng)注明Copyright (c) Duwen ,請(qǐng)支持原作,謝謝。

            posted on 2012-05-19 18:08 demons 閱讀(2614) 評(píng)論(0)  編輯 收藏 引用 所屬分類: The standard C++

            <2012年5月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久亚洲精品国产精品| 久久久久久亚洲精品无码| 欧美日韩久久中文字幕| 亚洲午夜久久久影院| 2022年国产精品久久久久| 久久国产精品一区二区| 色综合久久中文字幕综合网 | 亚洲AV乱码久久精品蜜桃| 久久久久成人精品无码中文字幕 | 久久精品国产2020| 久久66热人妻偷产精品9| 久久久久亚洲精品中文字幕| 久久精品国产亚洲AV蜜臀色欲| 国产精品99久久精品| 97香蕉久久夜色精品国产| 狠狠狠色丁香婷婷综合久久俺| 久久笫一福利免费导航 | 精品国际久久久久999波多野| 亚洲国产精品久久久久网站 | 国产精品热久久毛片| 久久免费的精品国产V∧| 久久人人爽人爽人人爽av | 久久精品国产免费| 无码人妻久久久一区二区三区 | 久久久久国产精品三级网| 久久精品国产久精国产思思| 99精品国产免费久久久久久下载| 精品久久人人妻人人做精品| 国产一区二区三区久久| 久久精品国产亚洲精品2020| 久久久久se色偷偷亚洲精品av| 久久影视综合亚洲| 久久精品人妻一区二区三区| 一本久久久久久久| 青青草国产精品久久| 办公室久久精品| 久久久受www免费人成| 久久影院亚洲一区| 国产精品久久久久免费a∨| 久久99九九国产免费看小说| 久久亚洲日韩看片无码|