自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C語言初學(xué)者常見問題與錯誤

開發(fā) 后端
不知不覺,學(xué)習(xí)C語言也快一年了。雖然有C語言課,但是老師完全讓我們自己看書,在自學(xué)的過程中,和周圍同學(xué)交流中,以及后來在 CSDN,BCCN,百度知道看帖回帖中,也看到許多C語言新人常遇到的問題與常犯的錯誤。為了讓新人們少走彎路少碰壁,我便打算寫下此文。當(dāng)然,由于我 自己的水平也有限,其中可能自己也不知不覺犯了錯誤,望高手們果斷斧正。

下面的程序,基本是摘自在CSDN,BCCN,百度知道的提問帖,以及我身邊的人和我自己寫的程序,限于篇幅,對于問題影響不大的部分我已經(jīng)刪去或改寫,一些與相應(yīng)問題無關(guān)的錯誤也一并進行了修改。對于完整的修正后的程序都在code::blocks 10.05(編譯器gcc,調(diào)試器gdb,平臺windows 7旗艦版)上運行過。

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語句進行判斷*/
    {
          printf("the Light is green,cars can run\n");  /*判斷結(jié)果為真時輸出*/
    }
    if(iSignal==0)         /*使用if語句進行判斷*/
    {
         printf("the Light is red,cars can't run\n");  /*判斷結(jié)果為真時輸出*/
    }
    return 0;
}

解析:在XP以上的系統(tǒng)用win-tc,dev-cpp等IDE編譯運行此程序時,無論輸入什么數(shù)字,結(jié)果都是一閃而過,因為程序執(zhí)行完語句后就直接退出了。

解決方案:這里有幾個辦法:

(1):在程序開頭加上#include,然后在程序末尾加上system("pause");
此方法僅適用于MS-DOS,windows,不適用于*nix等系統(tǒng)

(2):在需要暫停的地方加入一個或兩個getchar();
這種方法,實質(zhì)上并不是暫停程序,而是讓程序等待用戶輸入若干個回車。但是效果和暫停是相同的。

(3):在程序開頭加上#include,在需要暫停的地方加入一個getch();,原理和上一種差不多。在win-tc里用得比較多。

2、if,for,while的判斷后直接跟;

例如:

#include
int main(void)
{
    int a;
    scanf("%d",&a);
    if (a == 123);////錯誤,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是可以省略并可以正確運行的,但是這是非常不好的習(xí)慣。而main()這種寫法,C89標(biāo)準(zhǔn)勉強充許這種形式,C99標(biāo)準(zhǔn)是不允許的。而void main(),至今仍未有任何標(biāo)準(zhǔn)考慮接受它。但是有些編譯器的確允許。當(dāng)然,這種寫法廣為流行,應(yīng)該和老譚的書關(guān)系非常大。

解決方案:用到的頭文件應(yīng)該用include包含進去。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()將其解釋成,將鍵入一個數(shù)字,鍵入一個逗號,然后再鍵入一個數(shù)字,再鍵入一個逗號,最后再鍵入一個數(shù)學(xué)。也就是說必須像這樣輸入:2.3,5.1,3.8。如果不能精確匹配,則scanf()讀取將失敗。

作為編寫這個程序的人,你可以按照這個格式輸入,但是用戶則不知應(yīng)該以何種格式輸入。所以應(yīng)該改為scanf("%f%f%f",&a,&b,&c);

解決方案: scanf一行見解析。

5、scanf參數(shù)錯誤

例如:

#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)個參數(shù)里加上&的,因為scanf()函數(shù)里,第n(n>=2)個參數(shù)是變量的地址,而不是變量本身:例如定義int num;scnaf("%d", &num);而讀取字符串是不需要加上&,因為字符串的變量名本身就代表了地址。所以例子中應(yīng)為scanf("%s", str);同時,這條規(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為雙精度浮點型,而輸出使用%ld即長整型,數(shù)據(jù)類型不一致,輸出為0.PS:老譚的書講到用TC調(diào)試那一節(jié)舉的例子貌似就是int a; 后面寫到printf("%f",a);產(chǎn)生錯誤的。

解決方案:把b定義為長整型long,即long b = 1;(其實這里還涉及到隱式轉(zhuǎn)換,所以,更為正確的方法是把a也定義為長整型)

7、C語言中的“除法”

例如:

#include
int main(void)
{
    printf("請輸入一個華氏溫度\n");
    float a,c;
    scanf("%f",a);
    c=5/9*(a-32);
    printf("攝氏溫度為%4.2f",c);
    return 0;
}

解析: C語言中,兩個整型數(shù)相除,如果不能除盡,那么小數(shù)部分會直接被丟棄,即“截尾”。因此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;
}

解析:這段程序表面看起來沒有什么問題,但是,實際運行一遍的,就會發(fā)現(xiàn),只輸入了一組數(shù)據(jù),程序就退出了。
在開始的時候,程序運行良好,例如輸入 a 2,程序就會打印出aa。但是,程序還沒響應(yīng)第二次輸入就退出了。

問題就出在換行符,這次是緊跟在第一個輸入的2后面的那個換行符。scanf()函數(shù)將該換行符留在輸入隊列中,而getchar()并不跳過換行符。所以在循環(huán)的下一個周期,getchar()讀取了第一次輸入時的換行符,而換行符正是終止循環(huán)的條件。

解決方案:吃掉輸入流中的回車即可

在while循環(huán)最后,加上以下語句

while (getchar() != '\n')
    continue;

也可以加上fflush(stdin);刷新輸入流。

9、i++,i++;的糾結(jié)

例:

#include <stdio.h>

int main(void)
{
    int i = 5;
    printf("%d %d", i++, i++);
    return 0;
}

解析:這個問題,在實際編程應(yīng)該是沒有人會這么寫的,可還是有很多初學(xué)者糾結(jié)于此。在編程這個領(lǐng)域里,很多時候,實踐是最好的老師。對于代碼有疑問,那么上機敲一遍,編譯運行一遍是很好的方法。但是,在i++,i++這個方面,即使編譯運行了這個程序,也不一定會有正確的結(jié)果。用VC,TC,gcc編譯運行后的結(jié)果不一定相同?;蛘邚哪硞€角度來說,這里結(jié)果的正確與否其實并不重要了。

printf("%d %d", i++, i++);這一個語句,其中的i++,i++是未指定行為。即C語言的標(biāo)準(zhǔn)并沒有指定這運算是以何種順序進行的。如果用gcc編譯加上-Wall選項,那么會有warning:

gcc 2_1.c -o 2_1 -g -Wall
2_1.c: 在函數(shù)‘main’中:
2_1.c:6:24: 警告:‘i’上的運算結(jié)果可能是未定義的

因此,要解決這個問題,最好的方法就是不要在程序中寫這樣的代碼。

解決方案:見上。

10、60<=grade<=70

例:

if (60 <= grade <= 70)
   printf("及格");
else if(70 <= grade <= 85)
     printf("良好");

解析:在數(shù)學(xué)中,60<=grade<=70這種表達(dá)是成立的,但是在C語言中,并沒有這種表達(dá)。

解決方案:應(yīng)該改寫為

if ((60 <= grade) && (grade <= 70))
   printf("及格");
else ...

11、switch接受什么值?

例:

#include <stdio.h>

int main(void)
{
    double choice;
    scanf("%lf", &choice);
    switch(choice){
    case 1.0 : printf("1.0");
        break;
    case 2.0 : printf("2.0");
        break;
    default : printf("It's not 1.0 or 2.0");
    }
    return 0;
}

解析:這種問題同樣是不會出現(xiàn)在實際的編程當(dāng)中。但是一些C語言題目可能會這么出。switch()接受的是整數(shù):整型或者字符型。所以,浮點型,字符串等類型是不被switch()接受的。

解決方案:無他……

12、== =的困惑

例:

void count(void);/*計數(shù)器函數(shù)*/
void show_count(void);/*報告計數(shù)結(jié)果*/

int main(void)
{
    puts("Please enter some letters:(# to end)");
    count();
    show_count();

    puts("Thank you for using this program made by HerBal_Tea!");
    system("pause");
    return 0;
}

void count(void)
{
    while ((ch = getchar()) != STOP)
    {
        if (ch = SPACE)
        {
            sp_count++;
            continue;
        }

        if (ch = ENTER)
        {
            n_count++;
            continue;
        }

        other_count++;
    }/*end of while ((ch = getchar()) != '#')*/
}

void show_count(void)
{
    puts("The number of \nspace enter other");
    printf("%5d%6d%6d\n", sp_count, n_count, other_count);
}

解析:話說我自己也犯了這個錯。在數(shù)學(xué)中,=表示相等,而在很多編程語言中,=表示賦值,==才 表示相等。思維慣性導(dǎo)致錯誤。而且這個錯誤C編譯器既不會報錯也不會警告。出錯了檢查起來非常難。我看了不下二十遍調(diào)試的值監(jiān)視都找不到,最后還是一行一 行源代碼看,才知道是這個問題。在C語言四書五經(jīng)中的 c traps and pitfalls 和 expert c programming都提到過這個問題。

解決方案:無他,唯細(xì)心。

13、溢出問題

例:求斐波那契數(shù)列的前n項

#include <stdio.h>

#define N 100

int main(void)
{
    int fib[N];
    int i;

    fib[0] = fib[1] = 1;
    printf("%d %d ", fib[0], fib[1]);
    for (i = 2; i < N; ++i)
    {
        fib[i] = fib[i - 1] + fib[i - 2];
        printf("%d ", fib[i]);
    }
    return 0;
}

解析:在數(shù)學(xué)的概念中,整數(shù)、小數(shù)都是無限的,但是計算機中,即使是long long int或者double類 型都是有一定限度的,超過限度就會溢出。即是,假設(shè)一個指針式體重秤最大量程為120KG,那么一個體重130KG的人站上去,那么指針會指向10KG的 刻度處。溢出同理。而C語言,編譯器是不會對溢出進行檢查或者處理的。因此在編程中,應(yīng)該自己估算一下數(shù)的大小,以選用合適的數(shù)據(jù)類型來表示數(shù)據(jù)。

PS:由于TC編譯出的程序是16位的,所以int也是16位,很容易就會溢出。

解決方案:一方面是使用合適的數(shù)據(jù)類型,比如long ,long long 或者double。另一方面是,如果數(shù)特別大,連unsigned long long int或者連double都表示不了的時候,那么就用數(shù)組吧。不過定義數(shù)組的加減法還算好,但是定義乘除法就不是那么簡單了。所以推薦新人使用第一種方 法。第二種方法,可以考慮自己編個小程序?qū)崿F(xiàn)一下。

14、四舍五入?

例:要求將輸入的數(shù)按指定精度四舍五入輸出

#include <stdio.h>

int main(void)
{
    double a = 158.385427;
    printf("%.2lf", a);
    return 0;
}

解析: C語言中,printf("%.2lf", num);是直接截斷至小數(shù)點后兩位,并非四舍五入而是類似于趨零截尾。

解決方案:如下

//四舍五入顯示數(shù)字,精確由用戶輸入
//可辨別正負(fù) 
#include<stdio.h>
#include<stdlib.h>
#include<math.h>
int main(void)
{
    int width;
    double a = 158.385427;
    puts("Enter the width");
    scanf("%d",&width);
    if(a>0.0)
    a = (int)(a*pow(10,width) + 0.5)/pow(10,width);
    else 
    a = (int)(a*pow(10,width) - 0.5)/pow(10,width);
    printf("The result is %.*lf\n",width,a);
    system("pause");
    return 0;
}

15、函數(shù)的返回值

例:

#include<stdio.h>
double salary();/*計算工資稅的函數(shù)*/
void main()
{
    int choice;
    char want0;
    double result;
    while(1)
    {
        printf("歡迎使用個人所得稅計算器\n");
        printf("\n");
        printf("1.工資、薪金所得稅計算\n");
    printf("\n");
        printf("請輸入需要計算的稅收項目序號:");
        scanf("%d", &choice);
        if(choice==1)
        {
            salary();
            printf("%lf", result);
        }
double salary()
{
    double sal, result;
    printf("請輸入您的薪水:");
    scanf("%lf", &sal);
    if(sal<=3000)
    {
        result=sal;
    }
        if(sal>3000&&sal<=4500)
        {
            result=sal*0.05;
        }
        if(sal>4500&&sal<=7500)
        {
            result=sal*0.1-75;
        }
        if(sal>7500&&sal<=12000)
        {
            result=sal*0.2-525;
        }
        if(sal>12000&&sal<=38000)
        {
            result=sal*0.25-975;
        }
        if(sal>38000&&sal<=58000)
        {
            result=sal*0.3-2725;
        }
        if(sal>58000&&sal<=83000)
        {
            result=sal*0.35-5475;
        }
        if(sal>83000)
        {
            result=sal*0.45-13475;
        }
    return result;
}

解析:這個問題簡單來說是,用戶定義函數(shù)并未返回計算所得數(shù)值。往深的來說是關(guān)于存儲類、鏈接,即變量的作用域、鏈接、存儲時期的問題。

解決方案:

/*
 * main.c
 *
 * Created on: 2011-6-11
 * Author: ice
 */

#include<stdio.h>
double salary();/*計算工資稅的函數(shù)*/
int main(void)
{
    int choice;
//    char want0;////這個want0變量沒有使用
    double result;
    while(1)
    {
        printf("歡迎使用個人所得稅計算器\n");
        printf("\n");
        printf("1.工資、薪金所得稅計算\n");
     printf("\n");
        printf("請輸入需要計算的稅收項目序號:");
        scanf("%d", &choice);
        if(choice==1)
        {
            result = salary();//要有一個值接受返回值才可以
            printf("%lf", result);
        }
    }
}

double salary()
{
    double sal, result;
    printf("請輸入您的薪水:");
    scanf("%lf", &sal);
    if(sal<=3000)
    {
        result=sal;
    }
        if(sal>3000&&sal<=4500)
        {
            result=sal*0.05;
        }
        if(sal>4500&&sal<=7500)
        {
            result=sal*0.1-75;
        }
        if(sal>7500&&sal<=12000)
        {
            result=sal*0.2-525;
        }
        if(sal>12000&&sal<=38000)
        {
            result=sal*0.25-975;
        }
        if(sal>38000&&sal<=58000)
        {
            result=sal*0.3-2725;
        }
        if(sal>58000&&sal<=83000)
        {
            result=sal*0.35-5475;
        }
        if(sal>83000)
        {
            result=sal*0.45-13475;
        }
    return result;
}

 

 

責(zé)任編輯:王雪燕 來源: icelights
相關(guān)推薦

2011-08-24 09:54:05

Lua字符春交互

2013-02-20 10:13:38

JavaJava初學(xué)者

2022-10-19 23:18:27

KubernetesPod錯誤

2011-06-23 18:33:37

SEO

2010-01-14 17:18:17

C++語言

2025-04-07 01:35:00

Go語言程序

2010-03-10 17:09:26

Python語言

2011-06-08 09:37:19

C#Equals

2023-01-17 09:27:18

Python語言

2011-07-12 13:01:00

CC++

2011-04-12 10:13:24

2011-06-21 10:44:31

const

2010-03-15 18:55:22

Java編程語言

2011-09-16 09:38:19

Emacs

2022-04-24 15:21:01

MarkdownHTML

2022-07-22 15:55:32

Python代碼語言

2011-08-10 19:01:18

Objective-C

2009-12-24 17:26:00

ADO創(chuàng)建表

2010-01-15 19:05:42

學(xué)習(xí)C++

2009-11-18 09:30:43

點贊
收藏

51CTO技術(shù)棧公眾號