fgets函數及其用法,C語言fgets函數詳解

2020-07-16 10:04:23
雖然用 gets() 時有空格也可以直接輸入,但是 gets() 有一個非常大的缺陷,即它不檢查預留儲存區是否能夠容納實際輸入的資料,換句話說,如果輸入的字元數目大於陣列的長度,gets 無法檢測到這個問題,就會發生記憶體越界,所以程式設計時建議使用 fgets()。

fgets() 的原型為:

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

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

其中:s 代表要儲存到的記憶體空間的首地址,可以是字元陣列名,也可以是指向字元陣列的字元指標變數名。size 代表的是讀取字串的長度。stream 表示從何種流中讀取,可以是標准輸入流 stdin,也可以是檔案流,即從某個檔案中讀取,這個在後面講檔案的時候再詳細介紹。標準輸入流就是前面講的輸入緩衝區。所以如果是從鍵盤讀取資料的話就是從輸入緩衝區中讀取資料,即從標準輸入流 stdin 中讀取資料,所以第三個引數為 stdin。

下面寫一個程式:
# include <stdio.h>
int main(void)
{
    char str[20];  /*定義一個最大長度為19, 末尾是''的字元陣列來儲存字串*/
    printf("請輸入一個字串:");
    fgets(str, 7, stdin);  /*從輸入流stdin即輸入緩衝區中讀取7個字元到字元陣列str中*/
    printf("%sn", str);
    return 0;
}
輸出結果是:
請輸入一個字串:i love you
i love

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

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

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

下面寫一個程式看一下:
# include <stdio.h>
int main(void)
{
    char str[30];
    char *string = str;  //一定要先給指標變數初始化
    printf("請輸入字串:");
    fgets(string, 29, stdin);  //size指定為比字元陣列元素少一就行了
    printf("%s", string);  //printf中不需要新增'n', 因為字串中已經有了
    return 0;
}
輸出結果是:
請輸入字串:i love studying C語言
i love studying C語言

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

所以 fgets() 和 gets() 一樣,最後的回車都會從緩衝區中取出來。只不過 gets() 是取出來丟掉,而 fgets() 是取出來自己留著。但總之緩衝區中是沒有回車了!所以與 gets() 一樣,在使用 fgets() 的時候,如果後面要從鍵盤給字元變數賦值,那麼同樣不需要清空緩衝區。下面寫一個程式驗證一下。
# include <stdio.h>
int main(void)
{
    char str[30];
    char ch;
    printf("請輸入字串:");
    fgets(str, 29, stdin);
    printf("%s", str);  //後面不要加'n'
    scanf("%c", &ch);
    printf("ch = %cn", ch);
    return 0;
}
輸出結果是:
請輸入字串:i love you
i love you
Y
ch = Y