当前位置:网站首页>《C 陷阱与缺陷 》阅读概要
《C 陷阱与缺陷 》阅读概要
2022-08-04 13:26:00 【Aero Auto】
推荐指数:一星。
作为C语言经典书目,一直觉得有必要翻一番。但是读完之后大失所望,内容较少,也没有多少新意,并缺乏深度的视角。简单记录如下。
1.词法陷阱
1.1=与==的混淆
建议将左值放在前面,if (3==a)
1.2 $ 与$$,|与||
注意两者不要写错。
1.3 贪心法
每一个符号应该包含尽可能多的字符。也就是说,编译器将程序分解成符号的方法是,从左到右一个字符一个字符地读入,如果该字符可能组成一个符号,那么再读入下一个字符,判断已经读入的两个字符组成的字符串是否可能是一个符号的组成部分;如果可能,继续读入下一个字符,重复上述判断,直到读入的字符组成的字符串已不再可能组成一个有意义的符号。这个处理策略有时被称为“贪心法”,或者更口语化一点,称为“大嘴法”。
如 a—b意思是a-- - b,而不是a- --b;
a=-1在部分编译器上是a=a-1,而不是a = -1;
1.4 整型常量
如果一个整型常量的第一个字符是数字0,那么该常量将被视作八进制数。因此,10与010的含义截然不同。
有的编译器甚至支持八进制出现8以上的字符。
int a = 10;
int b = 010;
printf(" a:%d, b:%d\n",a,b);
其结果为: a:10, b:8
1.5 字符与字符串
“a”是a的ascii值加上\0的字符数组:97,0
‘a’是a的ascii值,就是97
2. 语法“陷阱”
2.1 优先级的设定
如果不确定运算符的优先级,请加上括号:
2.2 注意条件判断的的分号
if (3==a) ;
2.3 注意switch分支的break是否存在
没有break的分支会继续执行下一条case。
2.4 函数参数
即使函数不带参数,也应该包括参数列表。
func();执行函数
func是函数的地址,并不执行函数
2.5 if else的悬挂
注意else总是和最近的if配合,如果不确定,请加上{}.
3.语义陷阱
3.1 指针与数组
int array[4][10];
array[2]地址是array+2*10的位置。
3.2 malloc
注意malloc的返回值和free的配对使用
3.3 数组作为参数
将数组名作为参数,那么数组名会立刻被转换为指向该数组第1个元素的指针。因此,将数组作为函数参数毫无意义。所以,C语言中会自动地将作为参数的数组声明转换为相应的指针声明。
3.4 指针和指针指向的数据
复制指针并不同时复制指针所指向的数据。
3.5 空指针并非空字符串
3.6 注意数组边界,避免越界
3.7 求值顺序
即注意运算符优先级
3.8 运算符&&、|| 和 !
注意不要混用
3.9 整数溢出
3.10 main函数返回值
一个返回值为整型的函数如果返回失败,实际上是隐含地返回了某个“垃圾”整数。只要该数值不被用到,就无关紧要。
然而,大多数C语言实现都通过函数main的返回值来告知操作系统该函数的执行是成功还是失败。典型的处理方案是,返回值为0表示程序执行成功,返回值非0则表示程序执行失败。
因此,请注意给函数指定返回值!
4. 链接
4.1 链接器
推荐使用Lint对代码进行静态检测。
4.2 声明与定义
每个外部变量只能定义一次。
4.3 命名冲突与static修饰符
两个具有相同名称的外部对象实际上代表的是同一个对象。若干个函数需要共享一组外部对象,可以将这些函数放到一个源文件中,把它们需要用到的对象也都在同一个源文件中以static修饰符声明。
为了避免可能出现的命名冲突,如果一个函数仅仅被同一个源文件中的其他函数调用,我们就应该声明该函数为static。
4.4 形参、实参与返回值
如果一个函数在被定义或声明之前被调用,那么它的返回类型就默认为整型。
4.5 检查外部类型
外部变量的定义和引用时的声明类型必须一致。
4.6 头文件
每个外部对象只在一个地方声明。这个声明的地方一般就在头文件中,需要用到该外部对象的所有模块都应该包括这个头文件。特别需要指出的是,定义该外部对象的模块也应该包括这个头文件。
5 库函数
5.1 返回整数的getchar函数
getchar函数返回的是整型,不是char型。
5.2 更新顺序文件
文件操作时,注意当前的读写偏移。
一个输入操作不能随后直接紧跟一个输出操作,反之亦然。如果要同时进行输入和输出操作,必须在其中插入fseek函数的调用。
5.3 缓冲输出与内存分配
setbuf允许程序员在进行实际的写操作之前控制产生的输出数据量。注意buf实际的分配和释放时间,以免产生问题。
5.4 使用errno检测错误
与操作系统有关的库函数,当执行失败时会通过一个名称为errno的外部变量,通知程序该函数调用失败。
在调用库函数时,我们应该首先检测作为错误指示的返回值,确定程序执行已经失败,然后再检查errno,来搞清楚出错原因:
if open()<0
检查errno
5.5 库函数signal
一个信号可能在C程序执行期间的任何时刻发生。需要特别强调的是,信号甚至可能出现在某些复杂库函数(如malloc)的执行过程中。因此,从安全的角度考虑,信号的处理函数不应该调用上述类型的库函数。
6 预处理器
6.1 不能忽视宏定义中的空格
#define f (x) ((x)-1)
f后面有空格,因此f是表达 (x) ((x)-1)
6.2 宏并不是函数
我们最好在宏定义中把每个参数都用括号括起来。同样,整个结果表达式也应该用括号括起来,以防止当宏用于一个更大一些的表达式时可能出现的问题。
6.3 宏并不是语句
注意相关的括号,分号和逻辑的处理。
6.4 宏并不是类型定义
宏的一个常见用途是,使多个不同变量的类型可在一个地方说明:
但是使用typedef的方式要更加通用一些。这种情况下,我们尽量不使用宏。
7 可移植性缺陷
7.1 应对C语言标准变更
函数标准更新之后,是否需要向后保持兼容。
7.2 标识符名称的限制
注意大小写对编译器的区别,部分编译器不区分大小写。
7.3 整数的大小
short型、int型和long型在不同系统有差别,注意兼容性。
也许我们应该用typedef来重新设定各个整型的大小。
7.4 字符是有符号整数还是无符号整数
如果c是一个字符变量,使用(unsigned) c就可得到与c等价的无符号整数。这是会失败的,因为在将字符c转换为无符号整数时,c将首先被转换为int型整数,而此时可能得到非预期的结果。正确的方式是使用语句(unsigned char) c,因为一个unsigned char类型的字符在转换为无符号整数时无须首先转换为int型整数,而是直接进行转换。
不同系统可能有不同的差别。
7.5 移位运算符
使用移位运算符的程序员经常对以下两个问题感到困惑。
1.在向右移位时,空出的位是由0填充,还是由符号位的副本填充?
如果被移位的对象是无符号数,那么空出的位将被0填充。如果被移位的对象是有符号数,那么C语言实现既可以用0填充空出的位,也可以用符号位的副本填充空出的位。
2.移位计数(即移位操作的位数)允许的取值范围是什么?
如果被移位的对象长度是n位,那么移位计数必须大于或等于0,而严格小于n。因此,不可能做到在单次操作中将某个数值中的所有位都移出。为什么要有这个限制呢?因为只要加上了这个限制条件,我们就能够在硬件上高效地实现移位运算。
7.6 内存位置0
null指针并不指向任何对象。因此,除非是用于赋值或比较运算,出于其他任何目的使用null指针都是非法的。例如,如果p或q是一个null指针,那么strcmp(p, q)的值就是未定义的。
7.7 除法运算时发生的截断
假定我们让a除以b,商为q,余数为r :a/b=q,a%b=r。
我们希望a、b、q、r之间维持怎样的关系呢?
1.最重要的一点,我们希望q*b + r == a,因为这是定义余数的关系。
2.如果我们改变a的正负号,我们希望这会改变q的符号,但这不会改变q的绝对值。
3.当b>0时,我们希望保证r>=0且r<b。例如,如果余数用于哈希表的索引,确保它是一个有效的索引值很重要。
C语言或者其他语言在实现整数除法截断运算时,必须放弃上述3条原则中的至少一条。大多数程序设计语言选择了放弃第3条,而改为要求余数与被除数的正负号相同。
7.8 随机数的大小
ANSI C标准中定义了一个常数RAND_MAX,它的值等于随机数的最大取值,但是早期的C实现通常都没有包含这个常数。
7.9 大小写转换
库函数toupper和tolower也有与随机数类似的历史。它们起初被实现为宏,之后改成函数,最后又有新的宏。因此使用时可能会有区别。
7.10 首先释放,然后重新分配
注意remalloc的使用。
大多数C语言实现都为使用人员提供了3个内存分配函数:malloc、realloc和free。
UNIX系统参考手册第7版中描述的realloc函数的行为,略有不同。
早期的realloc函数的实现要求待重新分配的内存区域必须首先被释放。因为这个原因,仍然还有一些较老的C程序是首先释放某块内存,然后再重新分配这块内存。当移植这样一个较老的C程序到一个新的实现中时,我们必须注意到这一点。
7.11 可移植性问题的一个例子
该实例没有太大的参考意义。略过了。
边栏推荐
猜你喜欢
为什么密码云服务平台是云时代的必然之选?
《社会企业开展应聘文职人员培训规范》团体标准在新华书店上架
《社会企业开展应聘文职人员培训规范》团体标准在新华书店上架
ReentrantLock 原理
视觉SLAM十四讲学习笔记 第7讲 视觉里程计
Niuke.com Brush Question Record || Linked List
到底什么是真正的HTAP?
AutoCAD DWG,DXF文件导出高清图片、PDF
荧光磷脂PEG衍生物之一磷脂-聚乙二醇-荧光素,Fluorescein-PEG-DSPE
nVisual secondary development - Chapter 2 nVisual API operation guide Swagger use
随机推荐
《社会企业开展应聘文职人员培训规范》团体标准在新华书店上架
k8s上安装mysql
Script to get local IP address
【WeChat Mini Program】Social Internship Production Project for Information Management and Information System Major--Trash Fingerprint
烂大街的缓存穿透、缓存击穿和缓存雪崩,你真的懂了?
一文梳理NLP主要模型发展脉络
RobotFramework二次开发(一)
项目里的各种配置,你都了解吗?
LeetCode_3_无重复字符的最长子串
nVisual secondary development - Chapter 2 nVisual API operation guide Swagger use
【解决方案 三十一】Navicat数据库结构同步
Systemui qsSetting添加新图标
CReFF缓解长尾数据联邦学习(IJCAI 2022)
永磁同步电机FOC驱动代码讲解
Ultra-QuickSort
面试官:说一下NIO和BIO的区别
odoo13 note point
MySQL-数据类型
router---mode
【毕设选题推荐】机器人工程专业毕设选题推荐