2012軟件水平知識:程序員C語言新手入門

字號:

2012年軟件水平知識:程序員C語言新手入門
    不知不覺,學(xué)習(xí)C語言也快一年了。雖然有C語言課,但是老師完全讓我們自己看書,在自學(xué)的過程中,和周圍同學(xué)交流中,以及后來在CSDN,BCCN,百度知道看帖回帖中,也看到許多C語言新人常遇到的問題與常犯的錯(cuò)誤。為了讓新人們少走彎路少碰壁,我便打算寫下此文。當(dāng)然,由于我自己的水平也有限,其中可能自己也不知不覺犯了錯(cuò)誤,望高手們果斷斧正。
    下面的程序,基本是摘自在CSDN,BCCN,百度知道的提問帖,以及我身邊的人和我自己寫的程序,限于篇幅,對于問題影響不大的部分我已經(jīng)刪去或改寫,一些與相應(yīng)問題無關(guān)的錯(cuò)誤也一并進(jìn)行了修改。對于完整的修正后的程序都在code::blocks 10.05(編譯器gcc,調(diào)試器gdb,平臺(tái)windows 7旗艦版)上運(yùn)行過。
    1、程序畫面一閃而過
    #include
    int main(void)
    {
    int iSignal; /*定義變量表示信號燈的狀態(tài)*/
    printf(“the Red Light is 0,the Green Light is 1\n”); /*輸出提示信息*/
    scanf(“%d”,&iSignal); /*輸入iSignal變量*/
    if(iSignal==1) /*使用if語句進(jìn)行判斷*/
    {
    printf(“the Light is green,cars can run\n”); /*判斷結(jié)果為真時(shí)輸出*/
    }
    if(iSignal==0) /*使用if語句進(jìn)行判斷*/
    {
    printf(“the Light is red,cars can’t run\n”); /*判斷結(jié)果為真時(shí)輸出*/
    }
    return 0;
    }
    解析:在XP以上的系統(tǒng)用win-tc,dev-cpp等IDE編譯運(yùn)行此程序時(shí),無論輸入什么數(shù)字,結(jié)果都是一閃而過,因?yàn)槌绦驁?zhí)行完語句后就直接退出了。
    解決方案:這里有幾個(gè)辦法(1)
    在程序開頭加上#include
    然后在程序末尾加上system(“pause”);
    此方法僅適用于MS-DOS,windows,不適用于*nix等系統(tǒng)(2)
    在需要暫停的地方加入一個(gè)或兩個(gè)getchar();
    這種方法,實(shí)質(zhì)上并不是暫停程序,而是讓程序等待用戶輸入若干個(gè)回車。但是效果和暫停是相同的。(3)
    在程序開頭加上#include
    在需要暫停的地方加入一個(gè)getch();,原理和上一種差不多。在win-tc里用得比較多。
    2、if,for,while的判斷后直接跟;
    例如:
    #include
    int main(void)
    {
    int a;
    scanf(“%d”,&a);
    if (a == 123);////錯(cuò)誤,if(a == 123)后面不應(yīng)加上;,而應(yīng)該緊跟著
    ////{ printf();}代碼
    {
    printf(“ccc”);
    }
    else
    {
    printf(“ddd”);
    }
    getchar();
    return 0;
    }
    #include
    int main(void)
    {
    int n,i;
    printf(“please input a number》2:”);
    scanf(“%d”,&n);
    for(i=2;i
    {
    if(n%i==0)
    break;
    }
    if(i
    printf(“%d not a sushu\n”,n);
    else
    printf(“%d is a sushu\n”,n);
    return 0;
    }
    解析&解決方案:見注釋
    3、漏頭文件,main函數(shù)格式不規(guī)范。
    例如:
    main()
    {
    int a;
    scanf(“%d”,&a);
    printf(“input %d”, a);
    }
    解析:這段程序沒有帶上頭文件stdio.h。即漏寫了#include 。如果僅有scanf,printf函數(shù)的話,stdio.h是可以省略并可以正確運(yùn)行的,但是這是非常不好的習(xí)慣。而main()這種寫法,C89標(biāo)準(zhǔn)勉強(qiáng)充許這種形式,C99標(biāo)準(zhǔn)是不允許的。而void main(),至今仍未有任何標(biāo)準(zhǔn)考慮接受它。但是有些編譯器的確允許。當(dāng)然,這種寫法廣為流行,應(yīng)該和老譚的書關(guān)系非常大。
    解決方案:用到的頭文件應(yīng)該用include包含進(jìn)去。main()函數(shù)應(yīng)該寫成int main(void)這種形式,在main()函數(shù)尾部加上return 0;即
    #include
    int main(void)
    {
    int a;
    scanf(“%d”,&a);
    printf(“input %d”, a);
    return 0;
    }
    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è)逗號,然后再鍵入一個(gè)數(shù)字,再鍵入一個(gè)逗號,后再鍵入一個(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ī)則對于結(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即長整型,數(shù)據(jù)類型不一致,輸出為0.PS:老譚的書講到用TC調(diào)試那一節(jié)舉的例子貌似就是int a; 后面寫到printf(“%f”,a);產(chǎn)生錯(cuò)誤的。
    解決方案:把b定義為長整型long,即long b = 1;
    (其實(shí)這里還涉及到隱式轉(zhuǎn)換,所以,更為正確的方法是把a(bǔ)也定義為長整型)
    7、C語言中的“除法”
    例如:
    #include
    int main(void)
    {
    printf(“請輸入一個(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);刷新輸入流。