许多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&#61;1;a<&#61;6;aA&#43;&#43;)
b*&#61;A;
printf("%ld",b);
}
解析&#xff1a;定义b为双精度浮点型&#xff0c;而输出使用%ld即长整型&#xff0c;数据类型不一致&#xff0c;输出为0.PS&#xff1a;老谭的书讲到用TC调试那一节举的例子貌似就是int a; 后面写到printf("%f",a);产生错误的。
解决方案&#xff1a;把b定义为长整型long&#xff0c;即long b &#61; 1;(其实这里还涉及到隐式转换&#xff0c;所以&#xff0c;更为正确的方法是把a也定义为长整型)
7、C语言中的“除法”
例如&#xff1a;
#include
int main(void)
{
printf("请输入一个华氏温度\n"); float a,c;
scanf("%f",a);
c&#61;5/9*(a-32);
printf("摄氏温度为%4.2f",c); return 0;
}
解析&#xff1a;C语言中&#xff0c;两个整型数相除&#xff0c;如果不能除尽&#xff0c;那么小数部分会直接被丢弃&#xff0c;即“截尾”。因此5/9的结果是0.
解决方案&#xff1a;应该使用类型转换&#xff0c;或者明确相除的两数的类型
1.c&#61;(float)5/9*(a-32);
2.c&#61;5.0/9*(a-32);
3.c&#61;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 &#61; getchar()) !&#61; &#39;\n&#39;)
{
scanf("%d", &num);
for(i &#61; 0; i putchar(ch); putchar(&#39;\n&#39;); printf("Enter an another pair.Empty line to quit"); } return 0; } 解析&#xff1a;这段程序表面看起来没有什么问题&#xff0c;但是&#xff0c;实际运行一遍的&#xff0c;就会发现&#xff0c;只输入了一组数据&#xff0c;程序就退出了。在开始的时候&#xff0c;程序运行良好&#xff0c;例如输入a 2&#xff0c;程序就会打印出aa。但是&#xff0c;程序还没响应第二次输入就退出了。 问题就出在换行符&#xff0c;这次是紧跟在第一个输入的2后面的那个换行符。scanf()函数将该换行符留在输入队列中&#xff0c;而getchar()并不跳过换行符。所以在循环的下一个周期&#xff0c;getchar()读取了第一次输入时的换行符&#xff0c;而换行符正是终止循环的条件。 解决方案&#xff1a;吃掉输入流中的回车即可 在while循环最后&#xff0c;加上以下语句 while (getchar() !&#61; &#39;\n&#39;) continue; 也可以加上fflush(stdin);刷新输入流。 9、i&#43;&#43;,i&#43;&#43;;的纠结 例&#xff1a; #include int main(void) { int i &#61; 5; printf("%d %d", i&#43;&#43;, i&#43;&#43;); return 0; } 解析&#xff1a;这个问题&#xff0c;在实际编程应该是没有人会这么写的&#xff0c;可还是有很多初学者纠结于此。在编程这个领域里&#xff0c;很多时候&#xff0c;实践是最好的老师。对于代码有疑问&#xff0c;那么上机敲一遍&#xff0c;编译运行一遍是很好的方法。但是&#xff0c;在i&#43;&#43;,i&#43;&#43;这个方面&#xff0c;即使编译运行了这个程序&#xff0c;也不一定会有正确的结果。用VC&#xff0c;TC&#xff0c;gcc编译运行后的结果不一定相同。或者从某个角度来说&#xff0c;这里结果的正确与否其实并不重要了。 printf("%d %d", i&#43;&#43;, i&#43;&#43;);这一个语句&#xff0c;其中的i&#43;&#43;,i&#43;&#43;是未指定行为。即C语言的标准并没有指定这运算是以何种顺序进行的。如果用gcc编译加上-Wall选项&#xff0c;那么会有warning: gcc 2_1.c -o 2_1 -g -Wall 2_1.c:在函数‘main’中: 2_1.c:6:24:警告&#xff1a;‘i’上的运算结果可能是未定义的 因此&#xff0c;要解决这个问题&#xff0c;最好的方法就是不要在程序中写这样的代码。 解决方案&#xff1a;见上。 10、60<&#61;grade<&#61;70 例&#xff1a; if (60 <&#61; grade <&#61; 70) printf("及格"); else if(70 <&#61; grade <&#61; 85) printf("良好"); 解析:在数学中&#xff0c;60<&#61;grade<&#61;70这种表达是成立的&#xff0c;但是在C语言中,并没有这种表达。 解决方案&#xff1a;应该改写为 if ((60 <&#61; grade) && (grade <&#61; 70)) printf("及格"); else ... 11、switch接受什么值? 例&#xff1a; #include 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&#39;s not 1.0 or 2.0"); } return 0; } 解析&#xff1a;这种问题同样是不会出现在实际的编程当中。但是一些C语言题目可能会这么出。switch()接受的是整数&#xff1a;整型或者字符型。所以&#xff0c;浮点型&#xff0c;字符串等类型是不被switch()接受的。 解决方案&#xff1a;无他…… 12、&#61;&#61; &#61;的困惑 例&#xff1a; /**************************************** Copyright(C), 1999-2011, XX Co., Ltd. FileName:7121.c Author:……… Version:1.0 Time:23:06 Date:Feb 13th, 2011 Description:C Primer Plus编程练习7.12 1 Function List: 1.int main(void) 2.void count(void); 3.void show_count(void); History: ……… 23:06 2011/2/13 1.0 ****************************************/ #include #include #define STOP &#39;#&#39; /*以#为输入结束*/ #define SPACE &#39; &#39; /*定义空格*/ #define ENTER &#39;\n&#39; /*定义回车*/ /*空格计数器&#xff0c;换行符计数器&#xff0c;其他字符计数器&#xff0c;用户输入字符*/ int sp_count &#61; 0; int n_count &#61; 0; int other_count &#61; 0; char ch; 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 &#61; getchar()) !&#61; STOP) { if (ch &#61; SPACE) { sp_count&#43;&#43;; continue; } if (ch &#61; ENTER) { n_count&#43;&#43;; continue; } other_count&#43;&#43;; }/*end of while ((ch &#61; getchar()) !&#61; &#39;#&#39;)*/ } void show_count(void) { puts("The number of \nspace enter other"); printf("%5d%6d%6d\n", sp_count, n_count, other_count); } 解析&#xff1a;话说我自己也犯了这个错。在数学中&#xff0c;&#61;表示相等&#xff0c;而在很多编程语言中&#xff0c;&#61;表示赋值&#xff0c;&#61;&#61;才表示相等。思维惯性导致错误。而且这个错误C编译器既不会报错也不会警告。出错了检查起来非常难。我看了不下二十遍调试的值监视都找不到&#xff0c;最后还是一行一行源代码看&#xff0c;才知道是这个问题。在C语言四书五经中的c traps and pitfalls和expert c programming都提到过这个问题。 解决方案&#xff1a;无他&#xff0c;唯细心。 13、溢出问题 例&#xff1a;求斐波那契数列的前n项 #include #define N 100 int main(void) { int fib[N]; int i; fib[0] &#61; fib[1] &#61; 1; printf("%d %d ", fib[0], fib[1]); for (i &#61; 2; i { fib[i] &#61; fib[i - 1] &#43; fib[i - 2]; printf("%d ", fib[i]); } return 0; } 解析&#xff1a;在数学的概念中&#xff0c;整数、小数都是无限的&#xff0c;但是计算机中&#xff0c;即使是long long int或者double类型都是有一定限度的&#xff0c;超过限度就会溢出。即是&#xff0c;假设一个指针式体重秤最大量程为120KG&#xff0c;那么一个体重130KG的人站上去&#xff0c;那么指针会指向10KG的刻度处。溢出同理。而C语言&#xff0c;编译器是不会对溢出进行检查或者处理的。因此在编程中&#xff0c;应该自己估算一下数的大小&#xff0c;以选用合适的数据类型来表示数据。 PS&#xff1a;由于TC编译出的程序是16位的&#xff0c;所以int也是16位&#xff0c;很容易就会溢出。 解决方案&#xff1a;一方面是使用合适的数据类型&#xff0c;比如long ,long long或者double。另一方面是&#xff0c;如果数特别大&#xff0c;连unsigned long long int或者连double都表示不了的时候&#xff0c;那么就用数组吧。不过定义数组的加减法还算好&#xff0c;但是定义乘除法就不是那么简单了。所以推荐新人使用第一种方法。第二种方法&#xff0c;可以考虑自己编个小程序实现一下。 14、四舍五入? 例&#xff1a;要求将输入的数按指定精度四舍五入输出 #include int main(void) { double a &#61; 158.385427; printf("%.2lf", a); return 0; } 解析&#xff1a;C语言中&#xff0c;printf("%.2lf", num);是直接截断至小数点后两位&#xff0c;并非四舍五入而是类似于趋零截尾。 解决方案&#xff1a;如下 //四舍五入显示数字&#xff0c;精确由用户输入 //可辨别正负 #include #include #include int main(void) { int width; double a &#61; 158.385427; puts("Enter the width"); scanf("%d",&width); if(a>0.0) a &#61; (int)(a*pow(10,width) &#43; 0.5)/pow(10,width); else a &#61; (int)(a*pow(10,width) - 0.5)/pow(10,width); printf("The result is %.*lf\n",width,a); system("pause"); return 0; } 15、函数的返回值 例&#xff1a; #include double salary();/*计算工资税的函数*/ void main() { int choice; char want0; double result; while(1) { printf("欢迎使用个人所得税计算器\n"); printf("\n"); printf("1.工资、薪金所得税计算\n"); printf("\n"); printf("请输入需要计算的税收项目序号&#xff1a;"); scanf("%d", &choice); if(choice&#61;&#61;1) { salary(); printf("%lf", result); } double salary() { double sal, result; printf("请输入您的薪水&#xff1a;"); scanf("%lf", &sal); if(sal<&#61;3000) { result&#61;sal; } if(sal>3000&&sal<&#61;4500) { result&#61;sal*0.05; } if(sal>4500&&sal<&#61;7500) { result&#61;sal*0.1-75; } if(sal>7500&&sal<&#61;12000) { result&#61;sal*0.2-525; } if(sal>12000&&sal<&#61;38000) { result&#61;sal*0.25-975; } if(sal>38000&&sal<&#61;58000) { result&#61;sal*0.3-2725; } if(sal>58000&&sal<&#61;83000) { result&#61;sal*0.35-5475; } if(sal>83000) { result&#61;sal*0.45-13475; } return result; } 解析&#xff1a;这个问题简单来说是&#xff0c;用户定义函数并未返回计算所得数值。往深的来说是关于存储类、链接&#xff0c;即变量的作用域、链接、存储时期的问题。 解决方案&#xff1a; /* * main.c * * Created on: 2011-6-11 * Author: ice */ #include double salary();/*计算工资税的函数*/ int main(void) { int choice; // char want0;这个want0变量没有使用 double result; while(1) { printf("欢迎使用个人所得税计算器\n"); printf("\n"); printf("1.工资、薪金所得税计算\n"); printf("\n"); printf("请输入需要计算的税收项目序号&#xff1a;"); scanf("%d", &choice); if(choice&#61;&#61;1) { result &#61; salary();//要有一个值接受返回值才可以 printf("%lf", result); } } } double salary() { double sal, result; printf("请输入您的薪水&#xff1a;"); scanf("%lf", &sal); if(sal<&#61;3000) { result&#61;sal; } if(sal>3000&&sal<&#61;4500) { result&#61;sal*0.05; } if(sal>4500&&sal<&#61;7500) { result&#61;sal*0.1-75; } if(sal>7500&&sal<&#61;12000) { result&#61;sal*0.2-525; } if(sal>12000&&sal<&#61;38000) { result&#61;sal*0.25-975; } if(sal>38000&&sal<&#61;58000) { result&#61;sal*0.3-2725; } if(sal>58000&&sal<&#61;83000) { result&#61;sal*0.35-5475; } if(sal>83000) { result&#61;sal*0.45-13475; } return result; }