詳解C語(yǔ)言gets()函數(shù)與它的替代者fgets()函數(shù)
在c語(yǔ)言中讀取字符串有多種方法,比如scanf() 配合%s使用,但是這種方法只能獲取一個(gè)單詞,即遇到空格等空字符就會(huì)返回。如果要讀取一行字符串,比如:
I love BIT
這種情況,scanf()就無(wú)能為力了。這時(shí)我們最先想到的是用gets()讀取.
gets()函數(shù)從標(biāo)準(zhǔn)輸入(鍵盤(pán))讀入一行數(shù)據(jù),所謂讀取一行,就是遇到換行符就返回。gets()函數(shù)并不讀取換行符'\n',它會(huì)吧換行符替換成空字符'\0',作為c語(yǔ)言字符串結(jié)束的標(biāo)志。
gets()函數(shù)經(jīng)常和puts()函數(shù)配對(duì)使用,puts()函數(shù)用于顯示字符串,并自動(dòng)在字符串后面添加一個(gè)換行標(biāo)志'\n'
gets()函數(shù)存在一個(gè)嚴(yán)重的缺陷,這個(gè)缺陷就是:它不會(huì)檢查數(shù)組是否能夠裝的下輸入行:
比如:
我們定義了一個(gè)數(shù)組char src[5],這時(shí)候我們調(diào)用gets(src),來(lái)從標(biāo)準(zhǔn)輸入讀取字符串,我們看到gets()函數(shù)的參數(shù)為數(shù)組名,我們都知道,數(shù)組名就相當(dāng)于一個(gè)指針,也就是數(shù)組的首地址。這時(shí)如果我們的輸入大于5個(gè)字符,比如 I love BIT,gets()函數(shù)會(huì)從src所指地址開(kāi)始,依次填入每個(gè)字符,但是src只分配了5個(gè)字節(jié)的空間,填滿(mǎn)這五個(gè)空間后,gets()函數(shù)就會(huì)訪(fǎng)問(wèn)未被分配的內(nèi)存空間,如果這片空間已經(jīng)存有數(shù)據(jù),這時(shí)程序就會(huì)發(fā)生錯(cuò)誤,而中斷。
正式由于gets()函數(shù)的這個(gè)缺陷,在C99標(biāo)準(zhǔn)中,已經(jīng)不再建議使用gets()函數(shù),而在C11中更是直接拋棄了這個(gè)函數(shù)。
gets()被拋棄,那我們用什么來(lái)代替它的功能呢?
C11標(biāo)準(zhǔn)新增了gets_s()函數(shù)可以代替gets()函數(shù),但是,該函數(shù)是stdio.h輸入輸出函數(shù)系類(lèi)中的可選擴(kuò)展,因此,即使編譯器支持C11標(biāo)準(zhǔn),也有可能不支持gets_s()函數(shù)。
其實(shí)我們可以用c語(yǔ)言中的fgets()函數(shù)來(lái)代替gets()
我們先看一下函數(shù)原型聲明:
char *fgets(char *buf, int bufsize, FILE *stream);
注意一下第二個(gè)參數(shù)bufsize,這個(gè)參數(shù)就限制了讀取的字符的個(gè)數(shù),這就可以解決gets()函數(shù)的缺陷。
我們知道fgets() 函數(shù)主要用于讀取文件,如果要讀取鍵盤(pán),則stream參數(shù)應(yīng)該為stdin,
需要注意的是,如果bufsize設(shè)置為n,那么fgets()函數(shù)最多讀取n-1個(gè)字符,之所以用“最多”這個(gè)詞是因?yàn)?,如果在之前遇到了換行符,fgets函數(shù)也會(huì)返回。
還有一點(diǎn)就是,fgets()函數(shù)會(huì)讀取換行符(這一點(diǎn)和gets函數(shù)不同),當(dāng)讀取結(jié)束后,fgets函數(shù)會(huì)為buf在末尾添加一個(gè)空字符作為字符串的結(jié)束。
可以看一個(gè)簡(jiǎn)單的小例子:
#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); }
在這個(gè)程序中,我把數(shù)組的長(zhǎng)度設(shè)置為6,先看一組輸入和輸出:
輸入為zhan和回車(chē)('\n'),一共五個(gè)字符,fgets會(huì)讀取這五個(gè)字符,然后在末尾添加字符串結(jié)束標(biāo)志'\0';
我們知道fputs()函數(shù)并不會(huì)自動(dòng)添加換行,但是輸出結(jié)果卻換行輸出了Press any....,這就說(shuō)明了fgets()函數(shù)是會(huì)讀取換行符的。
在看一組輸入輸出:
這次我輸入了zhang和回車(chē)換行,fgets函數(shù)依然是讀取5個(gè)字符(LEN-1個(gè)),這時(shí)fgets()讀入zhang,已經(jīng)是五個(gè)字符了,所以回車(chē)換行并不會(huì)讀入,最后fgets()添加字符串結(jié)束標(biāo)志'\0',所以我們看到輸出時(shí),Press any...并沒(méi)有換行輸出,而是和zhang在同一行。
最后看一組輸入和輸出:
相信不用解釋大家也都明白了。
總結(jié)一下就是:
gets函數(shù)沒(méi)有限制讀入的個(gè)數(shù),這很可能會(huì)導(dǎo)致程序向未知的內(nèi)存空間寫(xiě)入數(shù)據(jù),而導(dǎo)致程序出錯(cuò)。
fgets函數(shù)中第二個(gè)參數(shù)限制了讀取的個(gè)數(shù),這也解決了gets函數(shù)存在的問(wèn)題,但要注意fgets函數(shù)只會(huì)讀取n-1個(gè)字符(如果遇到換行符會(huì)更少),并在最后添加字符串結(jié)束標(biāo)志,而且,fgets也會(huì)將換行符讀入。
以上所述是小編給大家介紹的詳解C語(yǔ)言gets()函數(shù)與它的替代者fgets()函數(shù)的相關(guān)知識(shí),希望對(duì)大家有所幫助,如果大家想了解更多,敬請(qǐng)關(guān)注腳本之家!
相關(guān)文章
如何使用C語(yǔ)言將數(shù)字、字符等數(shù)據(jù)寫(xiě)入、輸出到文本文件中
在分析數(shù)據(jù)時(shí),首先要解決數(shù)據(jù)的保存問(wèn)題,下面這篇文章主要給大家介紹了關(guān)于如何使用C語(yǔ)言將數(shù)字、字符等數(shù)據(jù)寫(xiě)入、輸出到文本文件中的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06淺析string 與char* char[]之間的轉(zhuǎn)換
與char*不同的是,string不一定以NULL('\0')結(jié)束。string長(zhǎng)度可以根據(jù)length()得到,string可以根據(jù)下標(biāo)訪(fǎng)問(wèn)。所以,不能將string直接賦值給char*2013-10-10C++實(shí)現(xiàn)string存取二進(jìn)制數(shù)據(jù)的方法
這篇文章主要介紹了C++實(shí)現(xiàn)string存取二進(jìn)制數(shù)據(jù)的方法,針對(duì)STL中string的用法進(jìn)行了較為詳細(xì)的分析,需要的朋友可以參考下2014-10-10C語(yǔ)言不用鏈表完成學(xué)生管理系統(tǒng)(完整代碼)
這篇文章主要介紹了C語(yǔ)言不用鏈表完成學(xué)生管理系統(tǒng)(完整代碼),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04C++實(shí)現(xiàn)管理系統(tǒng)的示例代碼
這篇文章主要介紹了C++實(shí)現(xiàn)管理系統(tǒng)的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10