程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> How to delete the specified line of a file in C++ on CUI con

How to delete the specified line of a file in C++ on CUI con

編輯:C++入門知識

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.
   昨天有人問我如何刪除文件中指定的一行(控制台下),我想這個問題在GUI下和CUI下還有點不同,我實驗了一下,感覺需要注意的地方還是挺多,所以我把心得整理一下,如有錯誤,歡迎指正。
   若在GUI下,我們在各種控件中編輯文本時,按回車換行時會產生兩個字符"\r\n", 而在CUI下則取決於我們寫入時的格式,當然如果是在CUI下,換行時我們當然不應該把用"\r\n"做換行符,因為我們完全可以只用一個'\n'就可以,沒必要再浪費多余一個字符了。當然本文給出的刪除指定行的方法同樣適用於"\r\n"換行的字符串,但本文主要針對CUI的。相反替換指定行的方法適用是有很多限制的,下面具體討論。
一: 刪除指定行
思路:兩種方法,一種是把文件指針定位到要刪除的行首,算出該行的長度,然後用空格替換掉每一個字符。另一種是把整個文件輸入一個字符數組,然後把它當作字符串處理,然後定位兩個位置,一個是要提換行的行首,另一個是下一行的行首,然後在把下一行到結束的所有字符平移到將要替換行的行首,也就是覆蓋法,用strcpy可輕松實現。
分析:若用第一種方法,優點是不用讀取整個文件(通常只需要讀到所要刪除行的下一行,下面代碼中有分析。)還有就是不用把所要刪除行的後面的全部文件內容整體向前平移。缺點就是不是真正的刪除,而是把要刪除行的內容用空格替換掉了,說以文件大小不會減小,當然這也是其致命弱點,但如果有替換某行內容的需要,那就是它了。第二種方法是真正刪除了指定行,因為要把文件內容全部讀入到一個字符數組,並且還要平移,所以性能上不如第一種,但它是真正的刪除了,還有就是第二種方法是把文件內容當作字符串來處理的,C++文件流輸出流在往文件寫字符串時是不帶結束符'\0'的,也就是如果在寫文件時沒有顯示寫'\0’,文件中是不會有'\0'的,也就是說當我們把一個文件所有字符讀入一個字符數組時,這個數組中是不會有'\0'的,說以在讀完文件內容後,我們得在結尾加一個'\0',這樣才能用字符串處理函數來處理。好了我們先來實現第二種方法。
delLine函數:
示例字符串:"aaaaa\nbbbbb\nccccc\n"
思路有了,我們來想想怎麼實現,我們還有一個問題沒有解決,就是如何確定要替換行的首末位置,比如現在要刪除第二行,我們則必須確定我用綠色標出的兩個位置,然後再把這兩個位置傳給strcpy,我們可以用字符查找函數strchr(字符串指針,'\n')找出這兩個位置,若找到,函數返回的是指向'\n'的指針,若沒找到,返回NULL,這裡要注意一個特例,就是如果我們要刪除第一行時,只需查找到第一個'\n'即可。好了,有了以說明,我寫了delLine函數,原型是void delLine(char* pFilepath , int nLine),第一個是文件路徑,第二個是要刪除的行,注意,在調用該函數前一定先要關閉掉打開該文件的所有流,這是系統文件共享的策略決定的,因為delLine中要對文件進行讀寫,如果調用該函數前,某個寫該文件的流沒關閉,這delLine函數中寫文件時會打開文件失敗,因為系統不用許同時多個用戶寫文件的。下面給出代碼:
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();
}
好了,下面實現方案一,方案一種的核心是確定要替換行的長度,用什麼方法呢? 想想,我是連續調用兩次getline函數,沒調用一次getline再緊接著調用tellg得到當前文件指針的文值,然後再用後一個位置減去前一個位置,這樣就得到了一個長度,這個長度並不是字符串的長度,來看這個字符串“aaaaa\nbbbbb\nccccc\n”,加入現在要替換第二行,第一次調用得到的位置是7,也就是第一個‘\n’後的一個字符位置,也就是下一行的首地址,第二次調用getline得到的位置是第一個c 的位置,也就是14,如果我們要替換的話只需保把5個b替換掉,顯然應該是14-7-2=5,說以我們得給兩次長度差再減2. 還需要注意的是但第二次調用getline時文件指針已經移向後一行了,所以得調用seekg調整文件指針。同樣要替換第一行時也是個特殊情況,只需調用一次getline就能確定第一行的長度。還有就是你要替換的行字符串必須大於或等於原行長度,當大於原行長度時,則超過的部分會被截斷,小於時,由於不能完全替換,應給出提示,所以調用時傳遞的新字符串寧長勿段,我們可以在有效字符後面加許多空格,這樣就萬事大吉了。下面給出代碼:
 
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();
}
好了,下面我們來進行測試:
把第一行替換成eeeee,第二行替換成f,注意此時由於f只有一個字符,所以根據寧長勿短原則,我們應該補上n個空格,在replaceLine 內部會自動截取原行長度個字符的,然後我們刪除第三行。
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();
}
輸出結果是:
eeeee
f
ddddd

 


摘自  不在浮沙築高台
 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved