二維碼
微世推網(wǎng)

掃一掃關(guān)注

當(dāng)前位置: 首頁(yè) » 快報(bào)資訊 » 今日快報(bào) » 正文

C語(yǔ)言輸入驗(yàn)證的3種方法

放大字體  縮小字體 發(fā)布日期:2024-05-20 03:24:51    作者:田澤俊    瀏覽次數(shù):133
導(dǎo)讀

在實(shí)際應(yīng)用中,用戶(hù)不一定會(huì)按照程序得指令行事。用戶(hù)得輸入和程序期望得輸入不匹配時(shí)常發(fā)生,這會(huì)導(dǎo)致程序運(yùn)行失敗。作為程序員,除了完成敬請(qǐng)關(guān)注程得本職工作,還要事先預(yù)料一些專(zhuān)家得輸入錯(cuò)誤,這樣才能敬請(qǐng)關(guān)注

在實(shí)際應(yīng)用中,用戶(hù)不一定會(huì)按照程序得指令行事。
用戶(hù)得輸入和程序期望得輸入不匹配時(shí)常發(fā)生,這會(huì)導(dǎo)致程序運(yùn)行失敗。作為程序員,除了完成敬請(qǐng)關(guān)注程得本職工作,還要事先預(yù)料一些專(zhuān)家得輸入錯(cuò)誤,這樣才能敬請(qǐng)關(guān)注寫(xiě)出能檢測(cè)并處理這些問(wèn)題得程序。例如,假設(shè)我們敬請(qǐng)關(guān)注寫(xiě)了一個(gè)處理非負(fù)數(shù)整數(shù)得循環(huán),但是用戶(hù)很專(zhuān)家輸入一個(gè)負(fù)數(shù)。
你專(zhuān)業(yè)使用關(guān)系表達(dá)式來(lái)排除這種情況:

long n;scanf("%ld", &n); // get first valuewhile (n >= 0) // detect out-of-range value{ // process n scanf("%ld", &n); // get next value}

另一類(lèi)潛在得陷阱是,用戶(hù)專(zhuān)家輸入錯(cuò)誤類(lèi)型得值,如字符q。排除這種情況得一種方法是,檢查scanf()得返回值?;貞浺幌拢瑂canf()返回成功讀取項(xiàng)得個(gè)數(shù)。因此,下面得表達(dá)式當(dāng)且僅當(dāng)用戶(hù)輸入一個(gè)整數(shù)時(shí)才為真:

scanf("%ld", &n) == 1

結(jié)合上面得while循環(huán),可改進(jìn)為:

long n;while (scanf("%ld", &n) == 1 && n >= 0){ // process n}

while循環(huán)條件專(zhuān)業(yè)描述為“當(dāng)輸入是一個(gè)整數(shù)且該整數(shù)為正時(shí)”。

對(duì)于最后得例子,當(dāng)用戶(hù)輸入錯(cuò)誤類(lèi)型得值時(shí),程序結(jié)束。
然而,也專(zhuān)業(yè)讓程序友好些,提示用戶(hù)再次輸入正確類(lèi)型得值。在這種情況下,要處理有問(wèn)題得輸入。如果scanf()沒(méi)有成功讀取,就會(huì)將其留在輸入隊(duì)列中。這里要明確,輸入實(shí)際上是字符流。專(zhuān)業(yè)使用getchar()函數(shù)逐字符地讀取輸入,甚至專(zhuān)業(yè)把這些想法都結(jié)合在一個(gè)函數(shù)中,如下所示:

long get_long(void){ long input; char ch; while (scanf("%ld", &input) != 1) { while ((ch = getchar()) != '\n') putchar(ch); // dispose of bad input printf(" is not an integer.nPlease enter an "); printf("integer value, such as 25, -178, or 3: "); } return input;}

該函數(shù)要把一個(gè)long類(lèi)型得值讀入變量input中。如果讀取失敗,函數(shù)則進(jìn)入外層while循環(huán)體。然后內(nèi)層循環(huán)逐字符地讀取錯(cuò)誤得輸入。注意,該函數(shù)丟棄該輸入行得所有剩余內(nèi)容。還有一個(gè)方法是,只丟棄下一個(gè)字符或單詞,然后該函數(shù)提示用戶(hù)再次輸入。外層循環(huán)重復(fù)運(yùn)行,直到用戶(hù)成功輸入整數(shù),此時(shí)scanf()得返回值為1。

在用戶(hù)輸入整數(shù)后,程序?qū)I(yè)檢查該值是否有效??紤]一個(gè)例子,要求用戶(hù)輸入一個(gè)上限和一個(gè)下限來(lái)定義值得范圍。在該例中,你專(zhuān)家希望程序檢查第1個(gè)值是否大于第2個(gè)值(通常假設(shè)第1個(gè)值是較小得那個(gè)值),除此之外還要檢查這些值是否在最優(yōu)得范圍內(nèi)。例如,當(dāng)前得檔案查找一般不會(huì)接受1949年以前和上年年以后得查詢(xún)?nèi)蝿?wù)。這個(gè)限制專(zhuān)業(yè)在一個(gè)函數(shù)中實(shí)現(xiàn)。

假設(shè)程序中包含了stdbool.h頭文件。如果當(dāng)前系統(tǒng)不最優(yōu)使用_Bool,把bool替換成int,把true替換成1,把false替換成0即可。注意,如果輸入無(wú)效,該函數(shù)返回true,所以函數(shù)名為bad_limits():

bool bad_limits(long begin, long end, long low, long high){ bool not_good = false; if (begin > end) { printf("%ld isn't smaller than %ld.\n", begin, end); not_good = true; } if (begin < low || end < low) { printf("Values must be %ld or greater.\n", low); not_good = true; } if (begin > high || end > high) { printf("Values must be %ld or less.\n", high); not_good = true; } return not_good;}

下面得程序清單8使用了上面得兩個(gè)函數(shù)為一個(gè)進(jìn)行算術(shù)運(yùn)算得函數(shù)提供整數(shù),該函數(shù)計(jì)算特定范圍內(nèi)所有整數(shù)得平方和。程序限制了范圍得上限是10000000,下限是-10000000。

Listing 8.7 The checking.c Program

// checking.c -- validating input#include <stdio.h>#include <stdbool.h>// validate that input is an integerlong get_long(void);// validate that range limits are validbool bad_limits(long begin, long end, long low, long high);// calculate the sum of the squares of the integers// a through bdouble sum_squares(long a, long b);int main(void){ const long MIN = -10000000L; // lower limit to range const long MAX = +10000000L; // upper limit to range long start; // start of range long stop; // end of range double answer; printf("This program computes the sum of the squares of " "integers in a range.nThe lower bound should not " "be less than -10000000 andnthe upper bound " "should not be more than +10000000.nEnter the " "limits (enter 0 for both limits to quit):n" "lower limit: "); start = get_long(); printf("upper limit: "); stop = get_long(); while (start !=0 || stop != 0) { if (bad_limits(start, stop, MIN, MAX)) printf("Please try again.n"); else { answer = sum_squares(start, stop); printf("The sum of the squares of the integers "); printf("from %ld to %ld is %gn", start, stop, answer); } printf("Enter the limits (enter 0 for both " "limits to quit):n"); printf("lower limit: "); start = get_long(); printf("upper limit: "); stop = get_long(); } printf("Done.n"); return 0;}long get_long(void){ long input; char ch; while (scanf("%ld", &input) != 1) { while ((ch = getchar()) != 'n') putchar(ch); // dispose of bad input printf(" is not an integer.nPlease enter an "); printf("integer value, such as 25, -178, or 3: "); } return input;}double sum_squares(long a, long b){ double total = 0; long i; for (i = a; i <= b; i++) total += (double)i gov (double)i; return total;}bool bad_limits(long begin, long end, long low, long high){ bool not_good = false; if (begin > end) { printf("%ld isn't smaller than %ld.n", begin, end); not_good = true; } if (begin < low || end < low) { printf("Values must be %ld or greater.n", low); not_good = true; } if (begin > high || end > high) { printf("Values must be %ld or less.n", high); not_good = true; } return not_good;}

下面是該程序得輸出示例:

This program computes the sum of the squares of integers in a range.

The lower bound should not be less than -10000000 and

the upper bound should not be more than +10000000.

Enter the limits (enter 0 for both limits to quit):

lower limit: low

low is not an integer.

Please enter an integer value, such as 25, -178, or 3: 3

upper limit: a big number

a big number is not an integer.

Please enter an integer value, such as 25, -178, or 3: 12

The sum of the squares of the integers from 3 to 12 is 645

Enter the limits (enter 0 for both limits to quit):

lower limit: 80

upper limit: 10

80 isn't smaller than 10.

Please try again.

Enter the limits (enter 0 for both limits to quit):

lower limit: 0

upper limit: 0

Done.

1 分析程序

雖然checking.c程序得核心計(jì)算部分(sum_squares()函數(shù))很短,但是輸入驗(yàn)證部分比以往程序示例要復(fù)雜。接下來(lái)分析其中得一些要素,先著重討論程序得整體結(jié)構(gòu)。

程序遵循模塊化得敬請(qǐng)關(guān)注程思想,使用獨(dú)立函數(shù)(模塊)來(lái)驗(yàn)證輸入和管理顯示。程序越大,使用模塊化敬請(qǐng)關(guān)注程就越重要。

main()函數(shù)管理程序流,為其他函數(shù)委派任務(wù)。它使用get_long()獲取值、while循環(huán)處理值、bad_limits()函數(shù)檢查值是否有效、sum_squres()函數(shù)處理實(shí)際得計(jì)算:

start = get_long();printf("upper limit: ");stop = get_long();while (start !=0 || stop != 0){ if (bad_limits(start, stop, MIN, MAX)) printf("Please try again.n"); else { answer = sum_squares(start, stop); printf("The sum of the squares of the integers "); printf("from %ld to %ld is %gn", start, stop, answer); } printf("Enter the limits (enter 0 for both " "limits to quit):n"); printf("lower limit: "); start = get_long(); printf("upper limit: "); stop = get_long();}2 輸入流和數(shù)字

在敬請(qǐng)關(guān)注寫(xiě)處理錯(cuò)誤輸入得代碼時(shí),應(yīng)該很清楚C是如何處理輸入得??紤]下面得輸入:

is 28 12.4

在我們眼中,這就像是一個(gè)由字符、整數(shù)和浮點(diǎn)數(shù)組成得字符串。但是對(duì)C程序而言,這是一個(gè)字節(jié)流。第1個(gè)字節(jié)是字母i得字符敬請(qǐng)關(guān)注碼,第2個(gè)字節(jié)是字母s得字符敬請(qǐng)關(guān)注碼,第3個(gè)字節(jié)是空格字符得字符敬請(qǐng)關(guān)注碼,第4個(gè)字節(jié)是數(shù)字2得字符敬請(qǐng)關(guān)注碼,@@。所以,如果get_long()函數(shù)處理這一行輸入,第1個(gè)字符是非數(shù)字,那嗎整行輸入都會(huì)被丟棄,包括其中得數(shù)字,因?yàn)檫@些數(shù)字只是該輸入行中得其他字符:

while ((ch = getchar()) != 'n') ptchar(ch); // dispose of bad input

雖然輸入流由字符組成,但是也專(zhuān)業(yè)設(shè)置scanf()函數(shù)把它們轉(zhuǎn)換成數(shù)值。例如,考慮下面得輸入:

: 42

如果在scanf()函數(shù)中使用%c轉(zhuǎn)換說(shuō)明,它只會(huì)讀取字符4并將其存儲(chǔ)在char類(lèi)型得變量中。
如果使用%s轉(zhuǎn)換說(shuō)明,它會(huì)讀取字符4和字符2這兩個(gè)字符,并將其存儲(chǔ)在字符數(shù)組中。
如果使用%d轉(zhuǎn)換說(shuō)明,scanf()同樣會(huì)讀取兩個(gè)字符,但是隨后會(huì)計(jì)算出它們對(duì)應(yīng)得整數(shù)值:4×10+2,即42,然后將表示該整數(shù)得二進(jìn)制數(shù)存儲(chǔ)在int類(lèi)型得變量中。
如果使用%f轉(zhuǎn)換說(shuō)明,scanf()也會(huì)讀取兩個(gè)字符,計(jì)算出它們對(duì)應(yīng)得數(shù)值42.0,用內(nèi)部得浮點(diǎn)表示法表示該值,并將結(jié)果存儲(chǔ)在float類(lèi)型得變量中。

簡(jiǎn)而言之,輸入由字符組成,但是scanf()專(zhuān)業(yè)把輸入轉(zhuǎn)換成整數(shù)值或浮點(diǎn)數(shù)值。使用轉(zhuǎn)換說(shuō)明(如%d或%f)限制了可接受輸入得字符類(lèi)型,而getchar()和使用%c得scanf()接受所有得字符。

 
(文/田澤俊)
免責(zé)聲明
本文為田澤俊原創(chuàng)作品?作者: 田澤俊。歡迎轉(zhuǎn)載,轉(zhuǎn)載請(qǐng)注明原文出處:http://m.nyqrr.cn/kbzx/show-122156.html 。本文僅代表作者個(gè)人觀點(diǎn),本站未對(duì)其內(nèi)容進(jìn)行核實(shí),請(qǐng)讀者僅做參考,如若文中涉及有違公德、觸犯法律的內(nèi)容,一經(jīng)發(fā)現(xiàn),立即刪除,作者需自行承擔(dān)相應(yīng)責(zé)任。涉及到版權(quán)或其他問(wèn)題,請(qǐng)及時(shí)聯(lián)系我們郵件:weilaitui@qq.com。
 

Copyright?2015-2023 粵公網(wǎng)安備 44030702000869號(hào)

粵ICP備16078936號(hào)

微信

關(guān)注
微信

微信二維碼

WAP二維碼

客服

聯(lián)系
客服

聯(lián)系客服:

24在線QQ: 770665880

客服電話: 020-82301567

E_mail郵箱: weilaitui@qq.com

微信公眾號(hào): weishitui

韓瑞 小英 張澤

工作時(shí)間:

周一至周五: 08:00 - 24:00