溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C語(yǔ)言scanf的工作原理是什么

發(fā)布時(shí)間:2022-01-04 17:15:07 來源:億速云 閱讀:240 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹“C語(yǔ)言scanf的工作原理是什么”,在日常操作中,相信很多人在C語(yǔ)言scanf的工作原理是什么問題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”C語(yǔ)言scanf的工作原理是什么”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!

原理解釋

先來觀察一段代碼和運(yùn)行結(jié)果:

#include <iostream>

using namespace std;

int main() {
    int a;
    char c;
    scanf("%d", &a);
    printf("a = %d", a);
    scanf("%c", &c);
    printf("c = %c", c);
}

C語(yǔ)言scanf的工作原理是什么

該代碼明明有兩個(gè) scanf ,但在運(yùn)行過程中,執(zhí)行完第一個(gè) scanf 和 printf 后,代碼直接停止了,并沒有繼續(xù)執(zhí)行下一個(gè) scanf ,這是為什么呢?

下面先介紹緩沖區(qū)原理。

行緩沖:在這種情況下,當(dāng)在輸入和輸出中遇到換行符時(shí),將執(zhí)行真正的IO操作。這時(shí),我們輸入的字符先存放到緩沖區(qū)中,等按下回車鍵換行時(shí)才進(jìn)行實(shí)際的IO操作.典型代表是標(biāo)準(zhǔn)輸入緩沖區(qū)(stdin)和標(biāo)準(zhǔn)輸出緩沖區(qū)(stdout)。

如上面例子所示,我們向標(biāo)準(zhǔn)輸人緩沖區(qū)中放入的字符為 &lsquo;20\n&rsquo;,輸入&rsquo;\n&rsquo;(回車)后, scanf 函數(shù)才開始匹配, scanf 函數(shù)中的%d 匹配整型數(shù) 20 ,然后放入變量 i 中,接著進(jìn)行打印輸出,這時(shí) &lsquo;\n&rsquo; 仍然在標(biāo)準(zhǔn)輸入緩沖區(qū)(stdin)內(nèi),如果第二個(gè) scanf 函數(shù)為 scanf("%d",&i) ,那么依然會(huì)發(fā)生阻塞,因?yàn)?scanf 函數(shù)在讀取整型數(shù)、浮點(diǎn)數(shù)、字符串(后面介紹數(shù)組時(shí)講解字符串)時(shí),會(huì)忽略 '\n&rsquo; (回車符)、空格符等字符(忽略是指scanf 函數(shù)執(zhí)行時(shí)會(huì)首先刪除這些字符,然后再阻塞), scanf 函數(shù)匹配一個(gè)字符時(shí),會(huì)在緩沖區(qū)刪除對(duì)應(yīng)的字符。因?yàn)樵趫?zhí)行 scanf("%c",&c) 語(yǔ)句時(shí),不會(huì)忽略任何字符,所以 scanf("%c",&c) 讀取了還在緩沖區(qū)中殘留的 &lsquo;\n&rsquo; 。

上面說的很專(啰)業(yè)(嗦),實(shí)際上就是:scanf 接收的是 %c,它把還存在緩沖區(qū)的 &lsquo;\n&rsquo; 當(dāng)成了一個(gè)字符,導(dǎo)致了代碼結(jié)束,如果 scanf 接收的是其他類型的數(shù)據(jù),則會(huì)忽略這個(gè) &lsquo;\n&rsquo;,繼續(xù)運(yùn)行下面的代碼,再舉一個(gè)例子:

#include <iostream>

using namespace std;

int main() {
    int a;
    int c;
    scanf("%d", &a);
    printf("a = %d", a);
    scanf("%d", &c);
    printf("c = %d", c);
}

C語(yǔ)言scanf的工作原理是什么

例如以上代碼,我輸入了好多個(gè)空格,但根本不影響實(shí)際的運(yùn)行結(jié)果,因?yàn)樗鼈兌急?printf 在緩沖區(qū)內(nèi)刪除掉了,scanf 是不會(huì)刪除緩沖區(qū)的內(nèi)容的。

再來看一段代碼理解一下:

#include <iostream>

using namespace std;

#define EOF (-1)

int main() {
    int i;
    while (scanf("%d", &i) != EOF) {
        printf("i=%d\n", i);
    }
}

C語(yǔ)言scanf的工作原理是什么

以上的 scanf 輸入,是 10,20,a 的順序輸入,在輸入 a 之后,代碼一直打印上一個(gè) printf 的內(nèi)容,這是因?yàn)椋?scanf 返回的是成功讀入的數(shù)據(jù)項(xiàng)數(shù),在我的輸入中輸入了一個(gè) a ,a 是無法匹配 %d 的,scanf 也不會(huì)刪除 a ,所以 scanf 的返回值是 0(沒有成功匹配),不等于 -1 ,此時(shí)就會(huì)一直 while 循環(huán)。

并且,在 scanf 返回值為 0 的情況下,沒有讀取 i 的值,此時(shí) i 的值還是上一次輸入的 20,這就會(huì)導(dǎo)致 while 循環(huán)一直打印上一次的 i=20。

C語(yǔ)言scanf的工作原理是什么

解決辦法

使用 rewind(stdin) 清空緩沖區(qū):

#include <iostream>

using namespace std;

#define EOF (-1)

int main() {
    int i;
    while (rewind(stdin), scanf("%d", &i) != EOF) {
        printf("i=%d\n", i);
    }
}

C語(yǔ)言scanf的工作原理是什么

到此,關(guān)于“C語(yǔ)言scanf的工作原理是什么”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI