软考程序员辅导:程序员c语言新人常见问题
不知不觉,学习C语言也快一年了。虽然有C语言课,但是老师完全让我们自己看书,在自学的过程中,和周围同学交流中,以及后来在CSDN,BCCN,百度知道看帖回帖中,也看到许多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);刷新输入流。