C语言易错点
C语言易错点考点(完整版)

总体上必须清楚的:1)程序结构是三种:顺序结构,循环结构(三个循环结构),选择结构(if和switch)2)用C语言编写的程序称为C语言源程序,源程序文件的后缀名为“.c”;源程序经编译后生成后缀名为“.obj”的目标文件;再把目标文件与各种库函数连接起来,生成“.exe”的可执行文件。
3)main函数,又称主函数,每个C程序有且只有一个main函数;无论main函数在整个程序中的位置如何,程序总是从main函数开始执行、也从main函数结束。
读程序时应从main 函数开始,然后从上往下读(碰到循环做循环,碰到选择做选择)。
4)计算机的数据在电脑中保存是以二进制的形式.数据存放的位置就是他的地址.5)bit是位是指为0或者1。
byte是指字节,一个字节=八个位.6)一定要记住二进制如何划成十进制。
概念常考到的:1、编译预处理不是C语言的一部分,不占运行时间,不要加分号。
C语言编译的程序称为源程序,它以ASCⅡ数值存放在文本文件中。
2、每个C语言程序中main函数是有且只有一个。
3、在函数定义中不可以再定义函数、即不允许函数嵌套定义。
4、算法可以没有输入,必须要有输出。
5、break可用于循环结构和switch语句。
6、逗号运算符的级别最低。
7、任意合法的表达式一定有一个数值与它对应。
第一章1)C程序中对字母的大小写是敏感的,如A与a是不同的标识符。
注释:是对程序的说明,目的是提高程序的可读性,可出现在程序中任意合适的地方,注释从“/*”开始到最近一个“*/”结束,其间任何内容都不会被计算机执行,不允许嵌套注释。
3)合法的用户标识符考查:合法的要求是由字母,数字,下划线组成。
有其它元素就错了。
并且第一个必须为字母或则是下划线。
第一个为数字就错了。
关键字不可以作为用户标识符号。
main define scanf printf都不是关键字。
迷惑你的地方If 是可以做为用户标识符。
因为If中的第一个字母大写了,所以不是关键字。
常见c语言错误(中英对照)

Ambiguous operators need parentheses 不明确的运算需要用括号括起Ambiguous symbol ''xxx''不明确的符号Argument list syntax error参数表语法错误Array bounds missing丢失数组界限符Array size toolarge数组尺寸太大Bad character in paramenters参数中有不适当的字符Bad file name format in include directive 包含命令中文件名格式不正确Bad ifdef directive synatax编译预处理ifdef有语法错Bad undef directive syntax编译预处理undef有语法错Bit field too large位字段太长Call of non-function调用未定义的函数Call to function with no prototype调用函数时没有函数的说明Cannot modify a const object不允许修改常量对象Case outside of switch漏掉了case 语句Case syntax errorCase 语法错误Code has no effect代码不可述不可能执行到Compound statement missing{分程序漏掉"{"Conflicting type modifiers不明确的类型说明符Constant expression required要求常量表达式Constant out of range in comparison在比较中常量超出范围Conversion may lose significant digits转换时会丢失意义的数字Conversion of near pointer not allowed 不允许转换近指针Could not find file ''xxx''找不到XXX文件Declaration missing ;说明缺少";"Declaration syntax error说明中出现语法错误Default outside of switchDefault 出现在switch语句之外Define directive needs an identifier定义编译预处理需要标识符Division by zero用零作除数Do statement must have whileDo-while语句中缺少while部分Enum syntax error枚举类型语法错误Enumeration constant syntax error枚举常数语法错误Error directive :xxx错误的编译预处理命令Error writing output file写输出文件错误Expression syntax error表达式语法错误Extra parameter in call调用时出现多余错误File name too long文件名太长Function call missing )函数调用缺少右括号Fuction definition out of place函数定义位置错误Fuction should return a value函数必需返回一个值Goto statement missing labelGoto语句没有标号Hexadecimal or octal constant too large 16进制或8进制常数太大Illegal character ''x''非法字符xIllegal initialization非法的初始化Illegal octal digit非法的8进制数字Illegal pointer subtraction非法的指针相减Illegal structure operation非法的结构体操作Illegal use of floating point非法的浮点运算Illegal use of pointer指针使用非法Improper use of a typedefsymbol类型定义符号使用不恰当In-line assembly not allowed不允许使用行间汇编Incompatible storage class存储类别不相容Incompatible type conversion不相容的类型转换Incorrect number format错误的数据格式Incorrect use of defaultDefault使用不当Invalid indirection无效的间接运算Invalid pointer addition指针相加无效Irreducible expression tree无法执行的表达式运算Lvalue required需要逻辑值0或非0值Macro argument syntax error宏参数语法错误Macro expansion too long宏的扩展以后太长Mismatched number of parameters in definition 定义中参数个数不匹配Misplaced break此处不应出现break语句Misplaced continue此处不应出现continue语句Misplaced decimal point此处不应出现小数点Misplaced elif directive不应编译预处理elifMisplaced else此处不应出现elseMisplaced else directive此处不应出现编译预处理else Misplaced endif directive此处不应出现编译预处理endif Must be addressable必须是可以编址的Must take address of memory location 必须存储定位的地址No declaration for function ''xxx''没有函数xxx的说明No stack缺少堆栈No type information没有类型信息Non-portable pointer assignment不可移动的指针(地址常数)赋值Non-portable pointer comparison不可移动的指针(地址常数)比较Non-portable pointer conversion不可移动的指针(地址常数)转换Not a valid expression format type不合法的表达式格式Not an allowed type不允许使用的类型Numeric constant too large数值常太大Out of memory内存不够用Parameter ''xxx'' is never used能数xxx没有用到Pointer required on left side of ->符号->的左边必须是指针Possible use of ''xxx'' before definition 在定义之前就使用了xxx(警告)Possibly incorrect assignment赋值可能不正确Redeclaration of ''xxx''重复定义了xxxRedefinition of ''xxx'' is not identical xxx的两次定义不一致Register allocation failure寄存器定址失败Repeat count needs an lvalue重复计数需要逻辑值Size of structure or array not known结构体或数给大小不确定Statement missing ;语句后缺少";"Structure or union syntax error结构体或联合体语法错误Structure size too large结构体尺寸太大Sub scripting missing ]下标缺少右方括号Superfluous & with function or array函数或数组中有多余的"&" Suspicious pointer conversion可疑的指针转换Symbol limit exceeded符号超限Too few parameters in call函数调用时的实参少于函数的参数不Too many default casesDefault太多(switch语句中一个)Too many error or warning messages错误或警告信息太多Too many type in declaration说明中类型太多Too much auto memory in function函数用到的局部存储太多Too much global data defined in file文件中全局数据太多Two consecutive dots两个连续的句点Type mismatch in parameter xxx参数xxx类型不匹配Type mismatch in redeclaration of ''xxx'' xxx重定义的类型不匹配Unable to create output file ''xxx''无法建立输出文件xxxUnable to open include file ''xxx''无法打开被包含的文件xxxUnable to open input file ''xxx''无法打开输入文件xxxUndefined label ''xxx''没有定义的标号xxxUndefined structure ''xxx''没有定义的结构xxxUndefined symbol ''xxx''没有定义的符号xxxUnexpected end of file in comment started on line xxx 从xxx行开始的注解尚未结束文件不能结束Unexpected end of file in conditional started on line xxx 从xxx 开始的条件语句尚未结束文件不能结束Unknown assemble instruction未知的汇编结构Unknown option未知的操作Unknown preprocessor directive: ''xxx''不认识的预处理命令xxxUnreachable code无路可达的代码Unterminated string or character constant字符串缺少引号User break用户强行中断了程序Void functions may not return a valueVoid类型的函数不应有返回值Wrong number of arguments调用函数的参数数目错''xxx'' not an argumentxxx不是参数''xxx'' not part of structurexxx不是结构体的一部分xxx statement missing (xxx语句缺少左括号xxx statement missing )xxx语句缺少右括号xxx statement missing ;xxx缺少分号xxx'' declared but never used说明了xxx但没有使用xxx'' is assigned a value which is never used给xxx赋了值但未用过Zero length structure结构体的长度为零。
C语言笔试易错知识点

C语言笔试易错知识汇总作者:毛志敏Email:wuxin0529@一、数据类型1、有符号数与无符号数的差异例如:unsigned int i;for(i=5; i>=0; i--){printf ( "%d ", i);}此题为一个死循环输出,但是大家往往没注意到无符号数,导致错误的认为输出是5 4 3 2 1 02、整型与浮点型的差异整型的除法与浮点型除法的差异浮点型不支持求余操作,应注意整型数据之间进行==、>、<操作,可直接拿两个数相比,但是浮点型数据由于精度问题却不能直接相比,如果需要判断两个浮点型数据是否相等,可以如下操作if ( abs(x-y) < 0.000000001)3、各类型混合运算如果一个运算符两边的运算数类型不同,先要将其转换为相同的类型,即较低类型转换为较高类型,然后再参加运算,转换规则如下图所示。
double ←── float 高↑long↑unsigned↑int ←── char,short 低不管最后结果如何,都会自动转换成赋值运算符左值的类型二、关键字1、switch case强烈提醒大家注意case后面有没有break,小心case穿透。
2、static声明静态变量,该类型变量在整个程序里只有一次初始化,然后便一直生存直到整个程序结束。
笔试中常有如下题目#include <stdio.h>int fun(int n){static int s = 0;s += n;return s;}void main(){int i;int sum = 0;for (i=0; i<5; i++){sum += fun(i);}printf("%d", sum);}此题目在主函数的for循环中多次调用fun函数,当第一次调用的时候,fun函数中的static变量s初始化为0,首次函数执行完后返回0(sum = 0);当第二次调用fum 函数时传入参数1,此时s已经存在,不需要初始化,函数返回1(sum = 1);同理第三次函数返回3(sum=4),第四次函数返回6(sum=10),第五次函数返回10(sum=20)。
「keilc语言编程常见错误分析」

1.Warning 280:’i’:unreferencedlocal variable 说明局部变量i 在函数中未作任何的存取操作解决方法消除函数中i 变量的宣告及即定义的参数在程序中并未调用2Warning 206:’Music3’:missing function-prototype 说明Music3( )函数未作宣告或未作外部宣告所以无法给其他函数调用解决方法将叙述void Music3(void)写在程序的最前端作宣告如果是其他文件的函数则要写成extern voidMusic3(void),即作外部宣告3Error:318:can’t open file‘beep.h’说明在编译C:\8051\MANN.C程序过程中由于main.c 用了指令#i nclude “beep.h”,但却找不到所致解决方法编写一个beep.h的包含档并存入到c:\8051 的工作目录中ﻫ4 Error237:’LedOn’:function already has a body ﻫ说明LedOn()函数名称重复定义即有两个以上一样的函数名称ﻫ解决方法修正其中的一个函数名称使得函数名称都是独立的ﻫﻫ5 ***WARNING16:UNCALLED SEGMENT,IGNORED FOR OVERLAYPROCESSSEGMENT: ?PR?_DELAYX1MS?DELAY说明DelayX1ms( )函数未被其它函数调用也会占用程序记忆体空间解决方法去掉DelayX1ms()函数或利用条件编译#if …..#endif,可保留该函数并不编译ﻫ6***WARNING6 :XDATASPACE MEMORY OVERLAPFROM : 0025HTO: 0025H ﻫ说明外部资料ROM的0025H 重复定义地址解决方法外部资料ROM 的定义如下Pdata unsigned char XF R_ADC _at_0x25 其中XFR_ADC 变量的名称为0x25,请检查是否有其它的变量名称也是定义在0x25 处并修正它ﻫ7 WARNING206:’DelayX1ms’:missingfunction-prototypeﻫC:\8051\INPUT.CError 267 :’DelayX1ms ‘:requires ANSI-styleprototypeC:\8051\INPUT.C说明程序中有调用DelayX1ms 函数但该函数没定义即未编写程序内容或函数已定义但未作宣告解决方法编写DelayX1ms的内容编写完后也要作宣告或作外部8宣告可在delay.h 的包含档宣告成外部以便其它函数调用ﻫﻫ***WARNING1:UNRESOLVED EXTERNAL SYMBOLﻫSYMBOL:MUSIC3解决办法:1.是文件没有添加到工程里。
大一C语言易错点总结和模拟试题带答案

易错点C语言中易犯的错误对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误.看着有错的程序,不知该如何改起,一些C编程时常犯的错误,写给各位学员以供参考.******************************** ******************************** ************************1.书写标识符时,忽略了大小写字母的区别.main(){int a=5;printf("%d",A);}编译程序把a和A认为是两个不同的变量名,而显示出错信息.C认为大写字母和小写字母是两个不同的字符.习惯上,符号常量名用大写,变量名用小写表示,以增加可读性.******************************** ******************************** ************************2.忽略了变量的类型,进行了不合法的运算.main(){float a,b;printf("%d",a%b);}%是求余运算,得到a/b的整余数.整型变量a和b可以进行求余运算,而实型变量则不允许进行"求余"运算.******************************** ******************************** ************************3.将字符常量与字符串常量混淆.char c;c="a";在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列.C规定以"\"作字符串结束标志,它是由系统自动加上的,所以字符串"a"实际上包含两个字符:'a'和'\',而把它赋给一个字符变量是不行的.******************************** ******************************** ************************4.忽略了"="与"=="的区别.在许多高级语言中,用"="符号作为关系运算符"等于".如在BASIC程序中可以写if (a=3) then …但C语言中,"="是赋值运算符,"=="是关系运算符.如:if (a==3) a=b;前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a.由于习惯问题,初学者往往会犯这样的错误.******************************** ******************************** ************************5.忘记加分号.分号是C语句中不可缺少的一部分,语句末尾必须有分号.a=1b=2编译时,编译程序在"a=1"后面没发现分号,就把下一行"b=2"也作为上一行语句的一部分,这就会出现语法错误.改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号.{z=x+y;t=z/100;printf("%f",t);}对于复合语句来说,最后一个语句中最后的分号不能忽略不写******************************** ******************************** ************************6.多加分号.对于一个复合语句,如:{z=x+y; t=z/100;printf("%f",t);};复合语句的花括号后不应再加分号,否则将会画蛇添足.又如:if (a%3==0);I++;本是如果3整除a,则I加1.但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1.再如:for (I=0;I<5;I++);{scanf("%d",&x);printf("%d",x);}本意是先后输入5个数,每输入一个数后再将它输出.由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它.****************************************************************************************7.输入变量时忘记加地址运算符"&".int a,b;scanf("%d%d",a,b);这是不合法的.Scanf函数的作用是:按照a,b在内存的地址将a,b的值存进去."&a"指a在内存中的地址.******************************** ******************************** ************************8.输入数据的方式与要求不符.①scanf("%d%d",&a,&b);输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:3,4输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab.②scanf("%d,%d",&a,&b);C规定:如果在"格式控制"字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符.下面输入是合法的:3,4此时不用逗号而用空格或其它字符是不对的.3 43:4又如:scanf("a=%d,b=%d",&a,&b); 输入应如以下形式:a=3,b=4****************************************************************************************9.输入字符的格式与要求不一致.在用"%c"格式输入字符时,"空格字符"和"转义字符"都作为有效字符输入.scanf("%c%c%c",&c1,&c2,&c3);如输入a b c字符"a"送给c1,字符" "送给c2,字符"b"送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔.****************************************************************************************10.输入输出的数据类型与所用格式说明符不一致.例如,a已定义为整型,b定义为实型a=3;b=4.5;printf("%f%d\n",a,b);编译时不给出出错信息,但运行结果将与原意不符.这种错误尤其需要注意.****************************************************************************************11.输入数据时,企图规定精度.scanf("%7.2f",&a);这样做是不合法的,输入数据时不能规定精度.****************************************************************************************12.switch语句中漏写break语句.例如:根据考试成绩的等级打印出百分制数段.switch(grade){ case 'A':printf("85~100\n"); case 'B':printf("70~84\n");case 'C':printf("60~69\n");case 'D':printf("<60\n"); default:printf("error\n");由于漏写了break语句,case只起标号的作用,而不起判断作用.因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二,三,四,五个printf函数语句.正确写法应在每个分支后再加上"break;".例如case 'C':printf("60~69\n");break; ****************************************************************************************13.忽视了while和do-while语句在细节上的区别.(1)main(){int a=0,I;scanf("%d",&I); while(I<=10){a=a+I;I++;}printf("%d",a);}(2)main(){int a=0,I;scanf("%d",&I);do{a=a+I;I++;}while(I<=10);printf("%d",a);}可以看到,当输入I的值小于或等于10时,二者得到的结果相同.而当I>10时,二者结果就不同了.因为while循环是先判断后执行,而do-while循环是先执行后判断.对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体.******************************** ******************************************************** 14.定义数组时误用变量. int n; scanf("%d",&n); int a[n]; 数组名后用方括号括起来的是常量表达式,可以包括常量和符号常 量.即C 不允许对数组的大小作动态定义. **************************************************************************************** 15.在定义数组时,将定义的"元素个数"误认为是可使的最大下 标值. main() {static int a[10]={1,2,3,4,5,6,7,8,9,10}; printf("%d",a[10]); } C 语言规定:定义时用a[10],表示a 数组有10个元素.其下标值由0 开始,所以数组元素a[10]是不存在的. **************************************************************************************** 16.初始化数组时,未使用静态存储. int a[3]={0,1,2}; 这样初始化数组是不对的.C 语言规定只有静态存储(static)数组 和外部存储 (exterm)数组才能初始化.应改为: static int a[3]={0,1,2}; **************************************************************************************** 17.在不应加地址运算符&的位置加了地址运算符. scanf("%s",&str); C 语言编译系统对数组名的处理是:数组名代表该数组的起始地址 ,且scanf 函数中的输入项是字符数组名,不必要再加地址符&.应 改为:scanf("%s",str); **************************************************************************************** 18.同时定义了形参和函数中的局部变量. int max(x,y) int x,y,z; {z=x>y?x:y; return(z); } 形参应该在函数体外定义,而局部变量应该在函数体内定义.应改 为: int max(x,y)int x,y;{int z;z=x>y?x:y;return(z);}试题一一、单项选择题(共30分,每题1分)1. 下列不正确的转义字符是A.\\B.\‘ C.074 D.\02. 不是C语言提供的合法关键字是A.switchB.cherC.caseD.default3.正确的标识符是()A.?a B.a=2 C.a.3 D.a_34.下列字符中属于键盘符号的是A.\ B.\n C.\t D.\b.下列数据中属于“字符串常量”的A.ABC B.“ABC” C.‘ABC’ D.‘A’6.char型常量在内存中存放的是A.ASCII码B.BCD码C.内码值D.十进制代码值7.设a为5,执行下列语句后,b 的值不为2的是()A.b=a/2B.b=6-(——a) C.b=a%2 D.b=a>3?2:28.在以下一组运算符中,优先级最高的运算符是()A.<= B.= C.% D.& &9.设整型变量i的值为3,则计算表达式i——i后表达式的值是()A.0B.1C. 2D.表达式出错10.设整型变量a,b,c均为2,表达式a+++b+++c++的结果是()A.6B.9C.8D.表达式出错11.若已定义x和y为double类型,则表达式x=1,y=x+3/2的值是()A.1B.2C.2.0D.2.512.设a=1,b=2,c=3,d=4,则表达式:a<b?a:c<d?a:d的结果是()A.4B.3C.2D.113.设a为整型变量,不能正确表达数学关系:10<a<15的C语言表达式是()A.10<a<15B.a==11 || a= =12 || a= =13 || a= =14C.a>10&&a<15D.!(a<=10)&&!(a>=15)14.若有以下定义:char a、 intb 、 floatc 、 double d,则表达式a*b+d-c值的类型为()A.floatB.intC.charD.double15.表达式“10!=9”的值是()A.trueB.非零值 C.0 D.116.循环语句 for (x=0,y=0;(y!=123)|| (x<4);x++);的循环执行()A.无限次B.不确定次 C.4次 D.3次17.在C语言中,下列说法中正确的是()A. 不能使用“do while”的循环B.“do while”的循环必须使用break语句退出循环C.“do while”的循环中,当条件为非0时将结束循环D.“do while”的循环中,当条件为0时将结束循环。
C语言的swap函数的易错点

C语⾔的swap函数的易错点程序⼀:交换值#include <stdio.h>void swap(int *x , int *y){int *temp;temp = x;x = y;y = temp;}void main(){int a = 1;int b = 2;swap(&a , &b);}对于程序⼀,在它运⾏完成之后,a,b的值并没有发⽣变化。
原因是swap函数⾥⾯的x,y都是形参,函数⾥⾯对形参的地址进⾏了交换,这并没有交换main函数中的a,b这两个变量指向的地址。
程序⼆:交换值#include <stdio.h>void swap(int *x , int *y){int *temp;temp = x;x = y;y = temp;}void main(){int *a = 1;int *b = 2;swap(a , b);}程序⼆也不能交换a,b所指向的值,原因类似于程序⼀的错误原因。
程序三:交换值#include <stdio.h>void swap(int x , int y){int temp;temp = x;x = y;y = temp;}void main(){int a = 1;int b = 2;swap(a , b);}程序三运⾏完之后,a,b的值也没有发⽣交换,是因为swap函数中的形参x,y发⽣了值的交换,⽽并不是main中实参的值交换。
程序四:交换字符串#include <stdio.h>void swap(char **x , char **y){char *temp;temp = *x;*x = *y;*y = temp;}void main(){char *a = "china";char *b = "hello";swap(&a , &b);}程序四可以交换两个字符串,其原理如下图所⽰:程序五:交换字符串#include <stdio.h>#include <string.h>void swap(char *x , char *y){char temp[10];strcpy(temp,x);strcpy(x,y);strcpy(y,temp);}void main(){char a[10] = "china";char b[10] = "hello";swap(a , b);}程序五也可以交换两个字符串,运⽤到了strcpy函数。
C语言面试易错点:负数整形的除法和取余

C语⾔⾯试易错点:负数整形的除法和取余C语⾔整形的除法的结果需要取整,例如5 / 3 = 1。
C语⾔中的取整⽅式是向零取整。
取整风格共有3种:1. 向零取整。
即向坐标轴0的⽅向,取最近的整数。
2. 向上取整。
也就是向+∞取整,即取不⼩于结果的最⼩整数。
3. 向下取整。
也就是向-∞取整,即取不⼤于结果的最⼤整数。
所以,C语⾔中计算-5除以3,因为实数结果约为-1.67,向零取整,得到结果为-1。
关于除法,⼈们常常⽤向右移位的⽅式来快速计算⼀个数除以2的幂的结果,例如16 / 4 = 4就是把10000右移2位得到100,即4;9 / 4 = 2就是把1001右移2位得到2。
事实上,这种快速计算的⽅法,是⼀种向下取整的过程,所以它只适⽤于被除数为正数的情况。
若被除数为负数,这种右移的快速计算⽅式就是错误的了。
下⾯⽤代码来验证⼀下,从10到-10,分别计算他们除以4和右移2位的结果:#include <stdio.h>int main(){int i = 0;for (i = 10; i >= -10; i--){printf("%d / 4 = %d\r\n", i, i / 4);printf("%d >> 2 = %d\r\n", i, i >> 2);printf("\r\n");}getchar();return 0;}程序运算结果如下图所⽰:可见,当被除数为正数的时候,除以4和右移2位的结果是⼀样的。
但是,当被除数为负数的时候,这两个结果并不⼀定相同。
关于负数的右移操作的解释:以-6为例,6的⼆进制形式是110,所以-6的补码形式是第⼀位符号位为1,数据位是110取反加1,得到1010。
1010右移2位,得到10。
取出10的符号位1,数据位是0,数据位取反加1得到10,即⼗进制的2。
也就是⼗进制-6右移2位得到⼗进制的-2。
C语言编程中常见的五种错误及对应解决方案

C语⾔编程中常见的五种错误及对应解决⽅案⽬录1. 未初始化的变量2. 数组越界3. 字符串溢出4. 重复释放内存5. 使⽤⽆效的⽂件指针前⾔:C 语⾔有时名声不太好,因为它不像近期的编程语⾔(⽐如 Rust)那样具有内存安全性。
但是通过额外的代码,⼀些最常见和严重的 C 语⾔错误是可以避免的。
即使是最好的程序员也⽆法完全避免错误。
这些错误可能会引⼊安全漏洞、导致程序崩溃或产⽣意外操作,具体影响要取决于程序的运⾏逻辑。
下⽂讲解了可能影响应⽤程序的五个错误以及避免它们的⽅法:1. 未初始化的变量程序启动时,系统会为其分配⼀块内存以供存储数据。
这意味着程序启动时,变量将获得内存中的⼀个随机值。
有些编程环境会在程序启动时特意将内存“清零”,因此每个变量都得以有初始的零值。
程序中的变量都以零值作为初始值,听上去是很不错的。
但是在 C 编程规范中,系统并不会初始化变量。
看⼀下这个使⽤了若⼲变量和两个数组的⽰例程序:#include <stdio.h>#include <stdlib.h>intmain(){int i, j, k;int numbers[5];int *array;puts("These variables are not initialized:");printf(" i = %d\n", i);printf(" j = %d\n", j);printf(" k = %d\n", k);puts("This array is not initialized:");for (i = 0; i < 5; i++) {printf(" numbers[%d] = %d\n", i, numbers[i]);}puts("malloc an array ...");array = malloc(sizeof(int) * 5);if (array) {puts("This malloc'ed array is not initialized:");for (i = 0; i < 5; i++) {printf(" array[%d] = %d\n", i, array[i]);}free(array);}/* done */puts("Ok");return 0;}这个程序不会初始化变量,所以变量以系统内存中的随机值作为初始值。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
表达式和基本语句
4、常量要放在 的左边。 常量要放在==的左边 常量要放在 的左边。 [示例]:这是大家比较习惯的写法。 if (data_type == RECT_AREA) { area_sum += rect_area[ind]; } 但是这是一种容易犯错的写法,为什么呢?你看下面这种错误: if (data_type = RECT_AREA) { area_sum += rect_area[ind]; } 看出不同了吗?就是“==”少了一个“=”号,这个错误很难查,编译器不会报错, 普通的调试和测试也不一定会出错,所以这种隐藏的错误很容易被漏过去,所以我 们采用下面这种写法: if (RECT_AREA == data_type) { area_sum += rect_area[ind]; }
表达式和基本语句
5、长语句要在低优先级操作符处划分新行,操作符放在新行之首,划分出的新行 长语句要在低优先级操作符处划分新行, 长语句要在低优先级操作符处划分新行 操作符放在新行之首, 要进行适当的缩进。使排版整齐,语句可读。 要进行适当的缩进。使排版整齐,语句可读。 [示例] 常用的写法: if ((taskno < max_act_task_number) && (n7stat_stat_item_valid (stat_item))) { ... // program code } 正确的写法: if ((taskno < max_act_task_number) && (n7stat_stat_item_valid (stat_item))) { ... // program code }
命名规则
命名规则可以说是比较复杂的,但也可以说是最简单的。为什么呢? 因为我们之所以要求命名符合规则,主要是为了让别人可以更方便的看懂你的代 码,但是要达成这个效果,遵守命名规则并不是唯一的方法,更简单的方法就是 加注释,只要你的注释可以让人清楚明白的知道这个变量的意义,那么也能达到 目的。 在这个方面,最容易犯错的就是这条规则: 变量命名要有具体含义外,还能表明变量类型、数据类型等, 变量命名要有具体含义外,还能表明变量类型、数据类型等,禁止取单个字符 ),但 等单个字符作为局部循环变量是允许的。 (如i、j、k…),但i、j、k等单个字符作为局部循环变量是允许的。 、、 ), 、 、 等单个字符作为局部循环变量是允许的 很多人在使用变量的时候都比较随意,特别是i、j、k这几个用的比较顺手的变量, 所以很多人都会不自觉的犯错,这个大家要注意。
排版规则
看了以上几段代码,有没有和你们自己的习惯相同的?如果有的话以后就需要改正 了。下面是正确的代码和规范: if (NULL == pUserCR) { return; } for (...) { ... // program code } if (...) { ... // program code } void example_fun( void ) { ... // program code }
排版规则
和空格相关的问题也是比较多的,因为这和很多人的编程习惯不同,所以有很多人都 会不自觉的犯错。 [示例]: (1) 逗号、分号只在后面加空格。 Fun(a, b, c); (2)比较操作符, 赋值操作符“=”、 “+=”,算术操作符“+”、“%”,逻辑操作符“&&”、“&”, 位域操作符"<<"、"^"等双目操作符的前后加空格。 if (current_time >= MAX_TIME_VALUE) a = b + c; a *= 2; a = b ^ 2; (3)"!"、"~"、"++"、"--"、"&"(地址运算符)等单目操作符前后不加空格。 *p = 'a'; // 内容操作"*"与内容之间 flag = !isEmpty; // 非操作"!"与内容之间 p = &mem; // 地址操作"&" 与内容之间 i++; // "++","--"与内容之间 (4)"->"、"."前后不加空格。 p->id = pid; // "->"指针前后不加空格 (5) if、for、while、switch等与后面的括号间应加空格,使if等关键字更为突出、明显。 if (a >= b && c > d)
变量、结构
5、若变量是字符串,需要将数组大小定义为指定长度加1的形式。 若变量是字符串,需要将数组大小定义为指定长度加 的形式 的形式。 若变量是字符串 [示例] 常用的写法 新增数组最大长度为32 readFileName[32]; 标准写法: 新增数组最大长度为32, 先定义一个长度的宏#define MAX_FILE_NAME_LEN 32, 然后定义参数readFileName[MAX_FILE_NAME_LEN + 1] 那么,为什么要用这种模式呢? 主要有两个优点,首先,使用宏定义,可以方便后期修改数组大小,只要修改宏定义 就可以修改所有相关的数据;其次,宏定义+1,为的就是保证数组在存放字符串的时 候有结束符。
C语言易错点
编制 时间 章浙锋 2011年07月21日
内容概述
排版 注释 命名规则 表达式和基本语句 变量、结构 函数、过程 内存管理 宏 其它
排版错误
排版主要是为了让代码整齐好看,有人说我写的代码我觉得挺好看的,而且自己 也写习惯了,为什么要按照规范来写? 因为在公司里不是一个人负责一套代码,而是大家合作,如果每个人都有自己的 风格,那么这套代码就会杂乱不堪,大家看起来很头痛,同时也影响工作效率。 排版方面的错误,最常见的是“{}”的相关错误。大家看下面几段代码: if (NULL == pUserCR) return; for (...) { ... // program code } if (...) { ... // program code } void example_fun( void ) { ... // program code }
注释错误
注释的错误有很多,最常犯的有3种: 1. 在函数头和源文件头要有注释 在函数头和源文件头要有注释; 2. 注释与所描述内容进行同样的缩进 注释与所描述内容进行同样的缩进; 3. 修改代码同时更新相应的注释,保证注释与代码的一致性。不再有用的注释要删 修改代码同时更新相应的注释,保证注释与代码的一致性。 除。 [示例]:如下例子,排版不整齐,阅读稍感不方便。 void example_fun( void ) { /* code one comments */ CodeBlock One } 应改为如下布局。 void example_fun( void ) { /* code one comments */ Co必须有 语句必须有default分支。 分支。 语句必须有 分支 3、写代码时,要注意表达式是否会上溢、下溢。 写代码时, 写代码时 要注意表达式是否会上溢、下溢。 [示例]:如下程序将造成变量下溢。 unsigned char size ; while (size-- >= 0) // 将出现下溢 { ... // program code } 当size等于0时,再减1不会小于0,而是0xFF,故程序是一个死循环。应如下修改。 char size; // 从unsigned char 改为char while (size-- >= 0) { ... // program code } 在以后的工作中,这个错误是经常会碰到的,特别是在修改别人的代码时,一不注 意就有可能出现这种错误,这个错误很难查,有时这个问题会卡你很久。
变量、结构
易出错的地方如下: 1、严禁局部变量与全局变量同名。 严禁局部变量与全局变量同名。 严禁局部变量与全局变量同名 2、严禁使用未经初始化的变量作为右值。 、严禁使用未经初始化的变量作为右值。 3、全局变量的读取 赋值应尽量放在一处处理,若多处使用应互斥保护。 赋值应尽量放在一处处理, 、全局变量的读取/赋值应尽量放在一处处理 若多处使用应互斥保护。 A、编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即P、V 操作)等手段对其加以保护。 B、构造仅有一个模块或函数可以修改、创建,而其余有关模块或函数只访问的 全局变量,防止多个不同模块或函数都可以修改、创建同一全局变量的现象。 C、当向全局变量传递数据时,要十分小心,防止赋与不合理的值或越界等现 象发生。 A、 semTake(globalMSem, WAIT_FOREVER); pDevCfgParam->periNum = ntohl(pManageCfg->periNum); semGive(globalMSem);
表达式和基本语句
表达式和基本语句中容易出错的地方不少,比如: 1、避免使用不易理解的数字,用有意义的标识来替代。涉及物理状态或者含有物 避免使用不易理解的数字, 避免使用不易理解的数字 用有意义的标识来替代。 理意义的常量,不应直接使用数字,必须用有意义的枚举或宏来代替。 理意义的常量,不应直接使用数字,必须用有意义的枚举或宏来代替。 [示例]:如下的程序可读性差。 if (0 == Trunk[index].trunk_state ) { Trunk[index].trunk_state = 1; ... // program code } 应改为如下形式: #define TRUNK_IDLE 0 #define TRUNK_BUSY 1 if (TRUNK_IDLE == Trunk[index].trunk_state ) { Trunk[index].trunk_state = TRUNK_BUSY; ... // program code }