1. 程式人生 > >c語言gets()函式與它的替代者fgets()函式

c語言gets()函式與它的替代者fgets()函式

在c語言中讀取字串有多種方法,比如scanf() 配合%s使用,但是這種方法只能獲取一個單詞,即遇到空格等空字元就會返回。如果要讀取一行字串,比如:

I love BIT

這種情況,scanf()就無能為力了。這時我們最先想到的是用gets()讀取.

gets()函式從標準輸入(鍵盤)讀入一行資料,所謂讀取一行,就是遇到換行符就返回。gets()函式並不讀取換行符'\n',它會吧換行符替換成空字元'\0',作為c語言字串結束的標誌。

gets()函式經常和puts()函式配對使用,puts()函式用於顯示字串,並自動在字串後面新增一個換行標誌'\n'

gets()函式存在一個嚴重的缺陷,這個缺陷就是:它不會檢查陣列是否能夠裝的下輸入行:

比如:

我們定義了一個數組char src[5],這時候我們呼叫gets(src),來從標準輸入讀取字串,我們看到gets()函式的引數為陣列名,我們都知道,陣列名就相當於一個指標,也就是陣列的首地址。這時如果我們的輸入大於5個字元,比如 I love BIT,gets()函式會從src所指地址開始,依次填入每個字元,但是src只分配了5個位元組的空間,填滿這五個空間後,gets()函式就會訪問未被分配的記憶體空間,如果這片空間已經存有資料,這時程式就會發生錯誤,而中斷。

正式由於gets()函式的這個缺陷,在C99標準中,已經不再建議使用gets()函式,而在C11中更是直接拋棄了這個函式。

gets()被拋棄,那我們用什麼來代替它的功能呢?

C11標準新增了gets_s()函式可以代替gets()函式,但是,該函式是stdio.h輸入輸出函式系類中的可選擴充套件,因此,即使編譯器支援C11標準,也有可能不支援gets_s()函式。

其實我們可以用c語言中的fgets()函式來代替gets()

我們先看一下函式原型宣告:

char *fgets(char *buf, int bufsize, FILE *stream);

注意一下第二個引數bufsize,這個引數就限制了讀取的字元的個數,這就可以解決gets()函式的缺陷。

我們知道fgets() 函式主要用於讀取檔案,如果要讀取鍵盤,則stream引數應該為stdin,

需要注意的是,如果bufsize設定為n,那麼fgets()函式最多讀取n-1個字元,之所以用“最多”這個詞是因為,如果在之前遇到了換行符,fgets函式也會返回。

還有一點就是,fgets()函式會讀取換行符(這一點和gets函式不同),當讀取結束後,fgets函式會為buf在末尾新增一個空字元作為字串的結束。

可以看一個簡單的小例子:

#include <stdio.h>
#include <stdlib.h>
#define LEN 6
int main(int argc,char* argv[])
{
    char src[LEN];
    printf("please enter:\n");
    fgets(src,LEN,stdin);
    printf("your enter is:\n");
    fputs(src,stdout);
}

在這個程式中,我把陣列的長度設定為6,先看一組輸入和輸出:

輸入為zhan和回車('\n'),一共五個字元,fgets會讀取這五個字元,然後在末尾新增字串結束標誌'\0';

我們知道fputs()函式並不會自動新增換行,但是輸出結果卻換行輸出了Press any....,這就說明了fgets()函式是會讀取換行符的

在看一組輸入輸出:

這次我輸入了zhang和回車換行,fgets函式依然是讀取5個字元(LEN-1個),這時fgets()讀入zhang,已經是五個字元了,所以回車換行並不會讀入,最後fgets()新增字串結束標誌'\0',所以我們看到輸出時,Press any...並沒有換行輸出,而是和zhang在同一行

最後看一組輸入和輸出:

相信不用解釋大家也都明白了。

總結一下就是:

gets函式沒有限制讀入的個數,這很可能會導致程式向未知的記憶體空間寫入資料,而導致程式出錯。

fgets函式中第二個引數限制了讀取的個數,這也解決了gets函式存在的問題,但要注意fgets函式只會讀取n-1個字元(如果遇到換行符會更少),並在最後新增字串結束標誌,而且,fgets也會將換行符讀入。