4、scanf格式控制誤用
例如:
#include
int main(void)
{
float a,b,c;
printf(“shuru 3 ge xi shu :”);
scanf(“%f,%f,%f”,&a,&b,&c);
printf(“he shi %f”,a + b + c);
return 0;
}
解析:scanf()函數(shù)允許把普通字符放在格式字符串中。除了空格字符之外的普通字符一定要與輸入串準(zhǔn)確匹配。否則,例如上面的程序,那么scanf()將其解釋成,將鍵入一個(gè)數(shù)字,鍵入一個(gè)逗號(hào),然后再鍵入一個(gè)數(shù)字,再鍵入一個(gè)逗號(hào),最后再鍵入一個(gè)數(shù)學(xué)。也就是說必須像這樣輸入:2.3,5.1,3.8。如果不能精確匹配,則scanf()讀取將失敗。
作為編寫這個(gè)程序的人,你可以按照這個(gè)格式輸入,但是用戶則不知應(yīng)該以何種格式輸入。所以應(yīng)該改為scanf(“%f%f%f”,&a,&b,&c);
解決方案:scanf一行見解析。
5、scanf參數(shù)錯(cuò)誤
例如:
#include
int main(void)
{
char str[80];
printf(“Please enter your first name”);
scanf(“%s”, &str);
printf(“Hello %s”, str);
return 0;
}
解析:scanf()中,讀取int, long, float, double, char等類型的數(shù)據(jù),是需要在第n(n>=2)個(gè)參數(shù)里加上&的,因?yàn)閟canf()函數(shù)里,第n(n>=2)個(gè)參數(shù)是變量的地址,而不是變量本身:例如定義int num;則scnaf(“%d”, &num);而讀取字符串是不需要加上&,因?yàn)樽址淖兞棵旧砭痛砹说刂。所以例子中?yīng)為scanf(“%s”, str);同時(shí),這條規(guī)則對(duì)于結(jié)構(gòu)體內(nèi)的變量的也適用,即
struct foo {
char ch;
char str[80];
int num
}data;
那么應(yīng)該是
scanf(“%c%s%d”, &data.ch, data.str, &data.num);
解決方案:見解析
6、數(shù)據(jù)類型混淆
例如:
#include
int main(void)
{ int a;
double b=1;
for(a=1;a<=6;aA++)
b*=A;
printf(“%ld”,b);
}
解析:定義b為雙精度浮點(diǎn)型,而輸出使用%ld即長(zhǎng)整型,數(shù)據(jù)類型不一致,輸出為0.PS:老譚的書講到用TC調(diào)試那一節(jié)舉的例子貌似就是int a; 后面寫到printf(“%f”,a);產(chǎn)生錯(cuò)誤的。
解決方案:把b定義為長(zhǎng)整型long,即long b = 1;
(其實(shí)這里還涉及到隱式轉(zhuǎn)換,所以,更為正確的方法是把a(bǔ)也定義為長(zhǎng)整型)
7、C語言中的“除法”
例如:
#include
int main(void)
{
printf(“請(qǐng)輸入一個(gè)華氏溫度\n”);
float a,c;
scanf(“%f”,a);
c=5/9*(a-32);
printf(“攝氏溫度為%4.2f”,c);
return 0;
}
解析:C語言中,兩個(gè)整型數(shù)相除,如果不能除盡,那么小數(shù)部分會(huì)直接被丟棄,即“截尾”。因此5/9的結(jié)果是0.
解決方案:應(yīng)該使用類型轉(zhuǎn)換,或者明確相除的兩數(shù)的類型
(1)c=(float)5/9*(a-32);
(2)c=5.0/9*(a-32);
(3)c=5.0/9.0*(a-32);
8、混合輸入數(shù)字和字符的杯具
#include
int main(void)
{
char ch;
int num, i;
printf(“Enter a character and a integer:\n”);
while((ch = getchar()) != ’\n‘)
{
scanf(“%d”, &num);
for(i = 0; i < num; ++i)
putchar(ch);
putchar(’\n’);
printf(“Enter an another pair.Empty line to quit”);
}
return 0;
}
解析:這段程序表面看起來沒有什么問題,但是,實(shí)際運(yùn)行一遍的,就會(huì)發(fā)現(xiàn),只輸入了一組數(shù)據(jù),程序就退出了。
在開始的時(shí)候,程序運(yùn)行良好,例如輸入 a 2,程序就會(huì)打印出aa。但是,程序還沒響應(yīng)第二次輸入就退出了。問題就出在換行符,這次是緊跟在第一個(gè)輸入的2后面的那個(gè)換行符。scanf()函數(shù)將該換行符留在輸入隊(duì)列中,而getchar()并不跳過換行符。所以在循環(huán)的下一個(gè)周期,getchar()讀取了第一次輸入時(shí)的換行符,而換行符正是終止循環(huán)的條件。
解決方案:吃掉輸入流中的回車即可
在while循環(huán)最后,加上以下語句
while (getchar() != ’\n‘)
continue;
也可以加上fflush(stdin);刷新輸入流。
北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |