【字符串】fgets函數及其用法詳解


雖然用 gets() 時有空格也可以直接輸入,但是 gets() 有一個非常大的缺陷,即它不檢查預留存儲區是否能夠容納實際輸入的數據,換句話說,如果輸入的字符數目大於數組的長度,gets 無法檢測到這個問題,就會發生內存越界,所以編程時建議使用 fgets()。

fgets() 的原型為:

1 #include<stdio.h>
2 char *fgets(char *c,int size,FILE *stream);

fgets() 雖然比 gets() 安全,但安全是要付出代價的,代價就是它的使用比 gets() 要麻煩一點,有三個參數。它的功能是從 stream 流中讀取 size 個字符存儲到字符指針變量 s 所指向的內存空間。它的返回值是一個指針,指向字符串中第一個字符的地址。

其中:s 代表要保存到的內存空間的首地址,可以是字符數組名,也可以是指向字符數組的字符指針變量名。size 代表的是讀取字符串的長度。stream 表示從何種流中讀取,可以是標准輸入流 stdin,也可以是文件流,即從某個文件中讀取,這個在后面講文件的時候再詳細介紹。標准輸入流就是前面講的輸入緩沖區。所以如果是從鍵盤讀取數據的話就是從輸入緩沖區中讀取數據,即從標准輸入流 stdin 中讀取數據,所以第三個參數為 stdin。

下面下一個程序:

 

1 #include <stdio.h>
2 int main(void)
3 {
4     char str[20];//定義一個最大長度為19,末尾是'\0'的字符數組來存儲字符串
5     printf("請輸入一個字符串:");
6     fgets(str,7,stdin);//從輸入流stdin即輸入緩沖區中讀取7個字符到字符數組str中
7     printf("%s\n",str);
8     return 0;
9 }

 

我們發現輸入的是“i love you”,而輸出只有“i love”。原因是 fgets() 只指定了讀取 7 個字符放到字符數組 str 中。“i love”加上中間的空格和最后的 '\0' 正好是 7 個字符。

那有人會問:“用 fgets() 是不是每次都要去數有多少個字符呢?這樣不是很麻煩嗎?”不用數!fget() 函數中的 size 如果小於字符串的長度,那么字符串將會被截取;如果 size 大於字符串的長度則多余的部分系統會自動用 '\0' 填充。所以假如你定義的字符數組長度為 n,那么 fgets() 中的 size 就指定為 n–1,留一個給 '\0' 就行了。

但是需要注意的是,如果輸入的字符串長度沒有超過 n–1,那么系統會將最后輸入的換行符 '\n' 保存進來,保存的位置是緊跟輸入的字符,然后剩余的空間都用 '\0' 填充。所以此時輸出該字符串時 printf 中就不需要加換行符 '\n' 了,因為字符串中已經有了。

下面寫一個程序看一下:

 1 # include <stdio.h>
 2 int main(void)
 3 {
 4     char str[30];
 5     //char *string = str;  //一定要先給指針變量初始化
 6     printf("請輸入字符串:");
 7     fgets(str, 29, stdin);  //size指定為比字符數組元素少一就行了 寫string時前面必須初始化
 8     printf("%s", str);  //printf中不需要添加'\n', 因為字符串中已經有了
 9     return 0;
10 }

我們看到,printf 中沒有添加換行符 '\n',輸出時也自動換行了。

所以 fgets() 和 gets() 一樣,最后的回車都會從緩沖區中取出來。只不過 gets() 是取出來丟掉,而 fgets() 是取出來自己留着。但總之緩沖區中是沒有回車了!所以與 gets() 一樣,在使用 fgets() 的時候,如果后面要從鍵盤給字符變量賦值,那么同樣不需要清空緩沖區。下面寫一個程序驗證一下。

 1 # include <stdio.h>
 2 int main(void)
 3 {
 4     char str[30];
 5     char ch;
 6     printf("請輸入字符串:");
 7     fgets(str, 29, stdin);
 8     printf("%s", str);  //后面不要加'\n' 會自動換行
 9     scanf("%c", &ch);
10     printf("ch = %c\n", ch);
11     return 0;
12 }
 

 


免責聲明!

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



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