C陷阱与缺陷——学习笔记

词法 “陷阱”

= 不同于 ==

符号=是赋值运算符,符号==是比较运算符

因为赋值运算出现频繁,所以使用较少的符号=来表示赋值操作.
而我们经常因为少打一个=号,在比较运算中使用赋值运算.

while(c = ' ' || c == '\t' || c == '\n')
    c = getc (f);

因为||的优先级高于赋值运算符=,所以上面表达式可以表示为

c = (' ' || c == '\t' || c == '\n')

因为' '的ASCII的值是32,所以c的值总是为1,while循环为死循环.

& 和 | 不同于 && 和 ||

& 和 | 是按位运算符
&& 和 || 是逻辑运算符

整形常量

如果一个整形常量的第一个字符是数字0,该常量被认为是八进制数.
所以01010含义完全不同.

字符与字符串

C语言中的单引号的字符表示一个整数,双引号表示字符串,执行字符的指针.

'a' 的长度为1
"a" 的长度为2   在内存空间中为  a \0
词法分析
  • 在用双引号括起的字符串中,注释符/* 属于字符串的一部分,而在注释中出现的双引号""又属于注释的一部分
  • 词法分析时: 每一个符号应该包含尽可能多的字符.(编译器将程序分解成符号的方法是,从左到右一个字符一个字符地读入,如果该字符可能组成一个符号,继续读入,知道读入的字符无法组成有意义的符号.)

语法 “陷阱”

理解函数声明

把声明中的变量名和声明末尾的分号去掉,再将剩余部分用括号括起来就得到了该类型的类型转换符

float (*h) ();
表示h是一个指向返回值为浮点类型的函数的指针
`(float (*)() )
表示一个指向返回值为浮点类型的函数的指针的类型转换符.

分析 (*(void(*)())0)()

  1. 假定变量fp是一个函数指针,调用fp所指向的函数方法是 (*fp)();
  2. 假设fp2是一个指向返回值为void类型的函数的指针,fp的声明为void (*fp2)();
  3. 根据声明对常熟0转型为指向返回值为void的函数的指针类型为(void(*)())0
  4. 最后我们可以用(void(*)())0来替换最初的fp得到
    (*(void(*)())0)();
  5. 我们可以使用typedef声明来简化问题为
typedef void (*funcptr)();
(* (funcptr)0)();
运算符的优先级问题

r = hi <<4 + low
因为加法运算的优先级高于移位运算,所以相当于
r = hi << (4 + low)

  1. 任何一个逻辑运算符的优先级低于任何一个关系运算符
  2. 移位运算符的优先级比算数运算符要低,但是比关系运算符要高.

[外链图片转存失败(img-hLYuRIN1-1564741494948)(…/images/C语言.md/2018-12-25-21-44-23.png)]

注意作为语句结束标志的分号

分号会结束ifwhile语句

if (x[i] > big);
    big = x[i];

因为if后面的分号,if语句会正常结束.所以等价于

if (x[i] > big){
}
    big = x[i];
函数调用

在C语言中,函数调用时即使函数不带参数,也应该包括参数列表.
如果f是一个函数.

f(); 是一个函数调用语句.
f; 什么也不做,这个语句计算了函数f的地址,并不调用该函数.

语义 “陷阱”

指针与数组

指针都是指向某种类型的变量.

如果a是一个数组,则*(a+1)是数组a中下标为1的元素的引用.*(a+i)是数组a中下标为i的元素的引用.所以a+i与i+a的含义相同,a[i]i[a]也具有相同的含义.

对于下面的声明
int calendar[12][31];
int *p;
int i;

我们可以通过
i = calendar[4][7];访问其中一个元素
我们也可以根据上面的用指针访问,如下
i =*(calendar[4]+7);
还可以进一步写成
i=*(*(calendar+4)+7);

我们再看 p =calendar;
这个语句是非法的,因为calendar是一个二维数组,就是"数组的数组",所以calendar是一个指向数组的指针,而p是一个指向整形变量的指针.这个语句试图将一种类型的指针赋值给另一种类型指针,所以是非法的

声明一个指向数组的指针.
int (*ap)[31];

作为参数的数组声明

在C语言中,如果我们使用数组名作为参数,数组名会被立刻转换为指向数组第一个元素的指针.

main (int argc,char* argv[]){
}
main(int argc,char **argv){
}

上面两种写法是完全相同的,第一种强调argv是一个指针数组,该数组的元素为字符指针类型.

边界计算与不对称边界

书中有一段这样的代码,非常经典

int i, a[10];
for (i=1; i<=10; i++)
    a[i] = 0;

因为数组越界a[10]其实修改的是i的值,所以当i=10的时候,a[10]会将i的值又修改为0,所以造成死循环.

求值顺序

因为求值顺序的原因,下面的代码无法正常工作

i = 0;
while (i < n)
    y[i++] = x[i];

因为我们无法确定y的值是自增后的还是自增前的值.
所以应该写为

for (i = 0; i < n; i++)
    y[i] = x[i];
main函数返回值

函数main与其他任何函数一样,如果并未显式声明返回类型,那么函数返回类型就默认为是整形.

连接

C语言的一个重要思想就是分别编译(Separate Compilation),即若干个源程序可以在不同的时候单独进行编译,然后在恰当的时候整合到一起.

声明与定义

extern int a 这个语句说明a是一个外部整型变量,但是因为它包括了extern关键字,显示地说明了a的存储空间是在程序的其他地方分配的.(对外部变量a的引用,而不是对a的定义.)

每个外部对象必须在程序某个地方进行定义
如果一个程序包含语句 extern int a;name这个程序就必须在某个地方包括int a对变量a的声明.

形参,实参与返回值
#include <stdio.h>
int main(int argc , char *argv[])
{
	int i;
	char c;
	for (i = 0; i < 5; i++)
	{
		scanf("%d", &c);
		printf("%d ", i);
	}
	printf("\n");
}

这段代码很有意思,我们声明c为char类型,而不是int类型.因为scanf函数被要求读入一个整数,应该传递一个指向整数的指针.而我们传入了一个指向字符的指针,scanf无法分辨这种情况.它只是将这个指向字符的指针当做指向整数的指针使用,并且在指针所指向的地方存储一个整数,因为整数占用4字节,而字符一个字节,所以会覆盖指针附近的内存空间.

当我们输入 305419896 1 2 3 4的时候,第一个数字的十六进制为(0x12345678).因为是整数所以他会覆盖掉变量i的低位的三字节.所以i的值变成了0x00123456>5,循环值循环了一次就会结束.最后输出的就是1193046(0x00123456)

库函数

缓冲输出与内存分配

程序输出有两种方式:1.即时处理方式2.先暂存起来,然后再大块写入的方式.

预处理器

getchar与putchar函数在实际中常常被定义为宏,这样可以避免处理字符时调用函数所需的运行时开销.

可移植性缺陷

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值