C語言中的gets()函數


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

比如:I love MIT

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

C標准庫中的 gets() 函數。

這個函數的定義如下:
char * gets ( char * str );

gets() 函數的形參只有一個指針。它會從標准輸入流中讀字符到一塊連續的內存地址空間中。這塊地址空間的開始位置就是指針 str 指向的位置。當在輸入流中遇到文件結束符( EOF )或者換行符(\n)時,讀取操作結束。當讀入換行符(\n)時,該字符不會被放入那塊連續的地址空間中。在讀取結束時, gets() 會自動在內存空間的末尾追加一個 NULL 字符。經過上述這些操作,對於程序員來說,這個函數得到的就是從標准輸入進來的,以 NULL 字符結尾的C字符串。如果讀入的字符流是一整行的話,行尾的換行符將會被舍去。

 

gets()函數從標准輸入(鍵盤)讀入一行數據,所謂讀取一行,就是遇到換行符就返回。

gets()函數並不讀取換行符'\n',它會把換行符替換成空字符'\0',作為c語言字符串結束的標志。

gets()函數經常和puts()函數配對使用,puts()函數用於顯示字符串,並自動在字符串后面添加一個換行標志'\n'

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

比如:

我們定義了一個數組char src[5],這時候我們調用gets(src),來從標准輸入讀取字符串,我們看到gets()函數的參數為數組名,我們都知道,數組名就相當於一個指針,也就是數組的首地址。這時如果我們的輸入大於5個字符,比如 I love MIT, gets()函數會從src所指地址開始,依次填入每個字符,但是src只分配了5個字節的空間,填滿這五個空間后,gets()函數就會訪問未被分配的內存空間,如果這片空間已經存有數據,這時程序就會發生錯誤,而中斷。

 

雖然 gets() 函數早就被公認為不安全的,但是它仍然存在於 C89 和 C99 標准,並最終在 C2011 標准中移除了。但這僅僅是在語言標准中的移除,當我檢查自己的一些代碼時,發現仍有地方用到了它。

 

除了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在末尾添加一個空字符作為字符串的結束。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM