第二條:一定要考慮除零、溢出、類型轉(zhuǎn)換、精度等細(xì)節(jié)問題!
雖然我們在平時學(xué)C語言的時候老師都提醒過,可是真正到了編程序的時候,卻沒有幾個人會考慮。這些細(xì)節(jié)問題似乎并不直接影響題目的輸出結(jié)果,然而,因為溢出和類型轉(zhuǎn)換錯誤而造成考試得不到滿分的可是大有人在啊! 我們看下面一道最害人的題目!,很多朋友考了這道題目都沒有過。
下面來看看你為什么是0分!
考試中心命題組64題:
請編制函數(shù)encryptChar(),按給定的替代關(guān)系對數(shù)組xx中的所有字符進行替代,
仍存入數(shù)組xx的對應(yīng)的位置上,最后調(diào)用函數(shù)WriteDat()把結(jié)果xx輸出到文件PS8.DAT中。
替代關(guān)系:f(p)=p*11 mod 256 (p是數(shù)組中某一個字符的ASCII值,f(p)是計算后新字符的ASCII值),
如果原字符小于等于32或大于130,則該字符不變,否則將f(p)所對應(yīng)的字符進行替代。
這道題目所有的人都做過,但幾乎就沒有人做對,包括南開無憂在內(nèi)!
原因就在于這一句!
:int f; f=(xx[i][k]*11)%6;
char *pf; *pf=*pf*11%6;
表面看來,好象兩個都對,
請你在TC下運行一下下面的代碼,看一下結(jié)果你就知道了是怎么回事了:
main()
{
char nk='E',*nkp,wuyou='E';
int wy;
nkp=&nk;
*nkp=*nkp*11%6;
wy=wuyou*11%6;
printf("%c %c %c %c\n",nk,*nkp,wuyou,wy);
printf("%d %d %d %d\n",nk,*nkp,wuyou,wy);
}
看到結(jié)果了么?是不是很吃驚啊?
-9 -9 69 247
大家看到了,南開的結(jié)果是負(fù)數(shù),連原來的變量nk都變成了負(fù)數(shù),這顯然錯了。
聰明一點,知道int比char大,定義了個int,對了。
錯誤原因就是在TC2.0中char型變量只有一個字節(jié),根本容納不下'E'*11,
所以產(chǎn)生了溢出,上次考這道題目沒過的同學(xué)是不是有所啟發(fā)呢?
這里給出考試中心命題組的答案:
void encryptChar(){
int i,k,ch;
for(i=0;i for(k=0;k ch=xx[i][k];
ch=(ch*11)%6;
if(!(ch<=32||ch>=130))xx[i][k]=ch;
}
}
似乎一個例子不能說明什么,那就把上面的數(shù)字題目的答案給大家看看:
考試中心命題組23題答案:
int i;
long val;
float dec;
for(i=0;i val=(long)xx[i];
aver+=xx[i]-val;
}
aver/=(double)N;
for(i=0;i val=(long)xx[i];
dec=xx[i]-val;
if(dec>aver){
sumint+=val;
sumdec+=(xx[i]-val);
}
}
看到了吧,首先,val的定義,不是我們習(xí)慣的int,而是能容納更大數(shù)的 long ,
這樣在求和的時候就不會溢出,無憂和南開的答案沒有吧
再看,算aver平均值的細(xì)節(jié),不是我們平常習(xí)慣的aver/=N,更不是無憂的1.0* sum/N,
而是aver/=(double)N,所以大家在考試的時候要時刻注意你的變量類型,特別是在一些求平均值的題目中,千萬要注意,用sum/n的時候,sum的類型一定不要用整形。
北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |