C语言初学者常见问题与错误

下面的程序,基本是摘自在CSDN,BCCN,百度知道的提问帖,以及我身边的人和我自己写的程序,限于篇幅,对于问题影响不大的部分我已经删去或改写,一些与相应问题无关的错误也一并进行了修改。对于完整的修正后的程序都在code::blocks 10.05(编译器gcc,调试器gdb,平台windows 7旗舰版)上运行过。

1、程序画面一闪而过

#include
int main(void)
{
    int iSignal;         /*定义变量表示信号灯的状态*/
    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");  /*判断结果为真时输出*/
    }
    if(iSignal==0)         /*使用if语句进行判断*/
    {
         printf("the Light is red,cars can't run\n");  /*判断结果为真时输出*/
    }
    return 0;
}

解析:在XP以上的系统用win-tc,dev-cpp等IDE编译运行此程序时,无论输入什么数字,结果都是一闪而过,因为程序执行完语句后就直接退出了。

解决方案:这里有几个办法:

(1):在程序开头加上#include,然后在程序末尾加上system("pause");
此方法仅适用于MS-DOS,windows,不适用于*nix等系统

(2):在需要暂停的地方加入一个或两个getchar();
这种方法,实质上并不是暂停程序,而是让程序等待用户输入若干个回车。但是效果和暂停是相同的。

(3):在程序开头加上#include,在需要暂停的地方加入一个getch();,原理和上一种差不多。在win-tc里用得比较多。

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

例如:

#include
int main(void)
{
    int a;
    scanf("%d",&a);
    if (a == 123);////错误,if(a == 123)后面不应加上;,而应该紧跟着
                  ////{ 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函数格式不规范

例如:

main()
{
    int a;
    scanf("%d",&a);
    printf("input %d", a);
}

解析:这段程序没有带上头文件stdio.h。即漏写了#include 。如果仅有scanf,printf函数的话,stdio.h是可以省略并可以正确运行的,但是这是非常不好的习惯。而main()这种写法,C89标准勉强充许这种形式,C99标准是不允许的。而void main(),至今仍未有任何标准考虑接受它。但是有些编译器的确允许。当然,这种写法广为流行,应该和老谭的书关系非常大。

解决方案:用到的头文件应该用include包含进去。main()函数应该写成int main(void)这种形式,在main()函数尾部加上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()函数允许把普通字符放在格式字符串中。除了空格字符之外的普通字符一定要与输入串准确匹配。否则,例如上面的程序,那么scanf()将其解释成,将键入一个数字,键入一个逗号,然后再键入一个数字,再键入一个逗号,最后再键入一个数学。也就是说必须像这样输入:2.3,5.1,3.8。如果不能精确匹配,则scanf()读取将失败。

作为编写这个程序的人,你可以按照这个格式输入,但是用户则不知应该以何种格式输入。所以应该改为scanf("%f%f%f",&a,&b,&c);

解决方案: scanf一行见解析。

5、scanf参数错误

例如:

#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等类型的数据,是需要在第n(n>=2)个参数里加上&的,因为scanf()函数里,第n(n>=2)个参数是变量的地址,而不是变量本身:例如定义int num;scnaf("%d", &num);而读取字符串是不需要加上&,因为字符串的变量名本身就代表了地址。所以例子中应为scanf("%s", str);同时,这条规则对于结构体内的变量的也适用,即

struct foo {
char ch;
char str[80];
int num
}data;

那么应该是

scanf("%c%s%d", &data.ch, data.str, &data.num);

解决方案:见解析

6、数据类型混淆

例如:

#include
int main(void)
{  int a;
   double b=1;
   for(a=1;a<=6;aA++)
      b*=A;
   printf("%ld",b);
}

解析:定义b为双精度浮点型,而输出使用%ld即长整型,数据类型不一致,输出为0.PS:老谭的书讲到用TC调试那一节举的例子貌似就是int a; 后面写到printf("%f",a);产生错误的。

解决方案:把b定义为长整型long,即long b = 1;(其实这里还涉及到隐式转换,所以,更为正确的方法是把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语言中,两个整型数相除,如果不能除尽,那么小数部分会直接被丢弃,即“截尾”。因此5/9的结果是0.

解决方案:应该使用类型转换,或者明确相除的两数的类型

  1. c=(float)5/9*(a-32);
  2. c=5.0/9*(a-32);
  3. c=5.0/9.0*(a-32);

8、混合输入数字和字符的杯具

#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;
}

解析:这段程序表面看起来没有什么问题,但是,实际运行一遍的,就会发现,只输入了一组数据,程序就退出了。
在开始的时候,程序运行良好,例如输入 a 2,程序就会打印出aa。但是,程序还没响应第二次输入就退出了。

问题就出在换行符,这次是紧跟在第一个输入的2后面的那个换行符。scanf()函数将该换行符留在输入队列中,而getchar()并不跳过换行符。所以在循环的下一个周期,getchar()读取了第一次输入时的换行符,而换行符正是终止循环的条件。

解决方案:吃掉输入流中的回车即可

在while循环最后,加上以下语句

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

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

9、i++,i++;的纠结

例:

#include <stdio.h>

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

解析:这个问题,在实际编程应该是没有人会这么写的,可还是有很多初学者纠结于此。在编程这个领域里,很多时候,实践是最好的老师。对于代码有疑问,那么上机敲一遍,编译运行一遍是很好的方法。但是,在i++,i++这个方面,即使编译运行了这个程序,也不一定会有正确的结果。用VC,TC,gcc编译运行后的结果不一定相同。或者从某个角度来说,这里结果的正确与否其实并不重要了。

printf("%d %d", i++, i++);这一个语句,其中的i++,i++是未指定行为。即C语言的标准并没有指定这运算是以何种顺序进行的。如果用gcc编译加上-Wall选项,那么会有warning:

gcc 2_1.c -o 2_1 -g -Wall
2_1.c: 在函数‘main’中:
2_1.c:6:24: 警告:‘i’上的运算结果可能是未定义的

因此,要解决这个问题,最好的方法就是不要在程序中写这样的代码。

解决方案:见上。

10、60<=grade<=70

例:

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

解析:在数学中,60<=grade<=70这种表达是成立的,但是在C语言中,并没有这种表达。

解决方案:应该改写为

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;
}

解析:这种问题同样是不会出现在实际的编程当中。但是一些C语言题目可能会这么出。switch()接受的是整数:整型或者字符型。所以,浮点型,字符串等类型是不被switch()接受的。

解决方案:无他……

12、== =的困惑

例:

void count(void);/*计数器函数*/
void show_count(void);/*报告计数结果*/

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);
}

解析:话说我自己也犯了这个错。在数学中,=表示相等,而在很多编程语言中,=表示赋值,==

表示相等。思维惯性导致错误。而且这个错误C编译器既不会报错也不会警告。出错了检查起来非常难。我看了不下二十遍调试的值监视都找不到,最后还是一行一
行源代码看,才知道是这个问题。在C语言四书五经中的 c traps and pitfalls 和 expert c
programming都提到过这个问题。

解决方案:无他,唯细心。

13、溢出问题

例:求斐波那契数列的前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;
}

解析:在数学的概念中,整数、小数都是无限的,但是计算机中,即使是long long int或者double

型都是有一定限度的,超过限度就会溢出。即是,假设一个指针式体重秤最大量程为120KG,那么一个体重130KG的人站上去,那么指针会指向10KG的
刻度处。溢出同理。而C语言,编译器是不会对溢出进行检查或者处理的。因此在编程中,应该自己估算一下数的大小,以选用合适的数据类型来表示数据。

PS:由于TC编译出的程序是16位的,所以int也是16位,很容易就会溢出。

解决方案:一方面是使用合适的数据类型,比如long ,long long
或者double。另一方面是,如果数特别大,连unsigned long long
int或者连double都表示不了的时候,那么就用数组吧。不过定义数组的加减法还算好,但是定义乘除法就不是那么简单了。所以推荐新人使用第一种方
法。第二种方法,可以考虑自己编个小程序实现一下。

14、四舍五入?

例:要求将输入的数按指定精度四舍五入输出

#include <stdio.h>

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

解析: C语言中,printf("%.2lf", num);是直接截断至小数点后两位,并非四舍五入而是类似于趋零截尾。

解决方案:如下

//四舍五入显示数字,精确由用户输入
//可辨别正负
#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、函数的返回值

例:

#include<stdio.h>
double salary();/*计算工资税的函数*/
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;
}

解析:这个问题简单来说是,用户定义函数并未返回计算所得数值。往深的来说是关于存储类、链接,即变量的作用域、链接、存储时期的问题。

解决方案:

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

#include<stdio.h>
double salary();/*计算工资税的函数*/
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;
}

来源:51CTO

时间: 2024-11-13 06:51:02

C语言初学者常见问题与错误的相关文章

c语言-C语言初学者提问,不会修改程序

问题描述 C语言初学者提问,不会修改程序 可以运行,但是达不到目的,不会修改 程序代码: typedef struct /* 学生结构 / { short status; / 数据状态,0:正常 1:删除 / char number[10]; / 学号 / char name[20]; / 姓名 / char sex[3]; / 性别 / int age; / 年龄 / char place; / 籍贯 / char department; / 系别 / char major; / 专业 / i

c-VS2013C语言代码无语法错误后调试窗口无显示结果的原因

问题描述 VS2013C语言代码无语法错误后调试窗口无显示结果的原因 include include int main() { int a[2],sum; a[1] = 0; a[0] = 35; sum = a[0] + a[1]; printf("%c", sum); return 0; } 解决方案 %c -> %d 在最后加上 getch() getchar() system("pause") 三者之一 解决方案二: 调试的时候,你设置断点了吗,是不是

关于do-while语句中while()判断条件--c语言初学者求助

问题描述 关于do-while语句中while()判断条件--c语言初学者求助 求两个正整数的最大公约数与最小公倍数如下程序 #include void main() { int m,n,a,b,i; scanf("%d%d",&m,&n); if(m>0&&n>0){ b=m*n; if(m<n) {a=m; m=n; n=a; } do{i=m%n; m=n; n=i;}while(n=0); printf("最大公约数为

代码-麻烦各位前辈看一下我这个C语言程序,找不出错误。

问题描述 麻烦各位前辈看一下我这个C语言程序,找不出错误. 我要对一个数组的数字进行排序,代码是这样的: //声明:该程序中输入的数据不能重复! #include int imin(const int va[], int n) // find the min of va[] { int i, min; min = va[0]; for (i = 0; i < n; i++) { if (min > va[i]) min = va[i]; } return min; } int imax(con

xcode-[c语言]无语法错误,但是结果不对,大一水平,求帮助~~!

问题描述 [c语言]无语法错误,但是结果不对,大一水平,求帮助--! 编译环境 xcode #include int main(int argc, const char * argv[]) { int k,i,j,a[4],m; printf("请输入4个不同的数字n"); for (i=0;i<4;i++) scanf("%d",&a[i]); for(i=0;i<4;i++) { for(j=0;j<4-i;j++) if(a[j]&g

内存-c语言调试出现总线错误,之前有出过非法指令,还有其他问题

问题描述 c语言调试出现总线错误,之前有出过非法指令,还有其他问题 还有这个my_connect 不能访问内存是什么情况,我之前也有碰到过,但是莫名其妙又好了,这次求大神一次性解决这些问题,定有重酬 解决方案 非法指令应该是内存没有分配,缓冲区溢出,把堆栈返回地址破坏了.这种问题不一定每次都出现,要仔细调试. 解决方案二: 只有这一小段,很难看出问题. 只有设断点,跟踪,看看问题出在那一段. 这类问题多出现在和指针.数组相关的地方. 解决方案三: buff[j*2 + 1] 这个地方,有没有越界

c语言-C语言初学者来求教,关于二分法

问题描述 C语言初学者来求教,关于二分法 原题是要求从一个有15个元素从大到小排列的数组中利用折半查找法查找到一个值x的位置,我写了如下代码,一运行就报错,求大神指正 在此谢过了! #include <stdio.h> main() { int num[15]={15,14,13,12,11,10,9,8,7,6,5,4,3,2,1}; int m=0,n=14,x,p,q; printf("请输入欲定位的数x "); scanf("%d",x); wh

c语言-一个C语言初学者的疑问

问题描述 一个C语言初学者的疑问 我是一个初学者,希望各位大神能帮我看看这串代码错在哪了,我用的IDE是VS2013,编写C语言代码,拜托了 #include #define N 3 struct Student { int num; char name[20]; float score[3]; float aver; }; int main() { void input(struct Student stu[]); struct Student max(struct Student stu[]

请问大神错在哪里 c语言初学者

问题描述 请问大神错在哪里 c语言初学者 #includeint main(void)one-three(void);two(void);{ printf(""starting now.n"");one-three();two();printf(""three.n"");printf(""done!n"");return 0; }one-three(void){ printf("