词法分析中的“贪心法”
- 每个符号应该包含尽可能多的字符。
a---b
等同于(a--)-b
。
a+++++b
被编译器视为((a++)++)+b
,语法上是不正确的。
字符与字符串
单引号引起的字符代表一个整数。
双引号引起的字符串代表一个指向无名数组起始字符的指针。
理解函数声明
(* (void (*) () ) 0) ();
这是一个经典的例子。分析如下:
void (*) ()
指向返回值为void类型的函数指针( void (*) () ) 0
将0进行类型转换(* ( void (*) () ) 0) ();
调用地址为0的子例程
运算符优先级
- 优先级由高到低排列
- 数组下标** [ ]、函数调用操作符( )、各结构成员选择操作符 ->**和 **. **。自左与右结合。
- 单目运算操作符,包括类型转换。 自右至左结合。
- 双目运算操作符,其中优先级顺序 算术运算符>位移运算符>关系运算符>逻辑运算符。 自左向右结合
- 三目运算符(条件运算符)。 自右至左结合。
- 赋值运算符。 自右至左结合。
- 逗号运算符。 自左与右结合。
运算符优先级所引发的错误很难发现。
while ( c == '\t '|| c = ' '|| c == '\n' )
事实上,该语句的正确划分如下。
while ( (c == '\t '|| c ) = ( ' '|| c == '\n' ))
该语句不仅是将==误写为=,还导致了表达式左端为不可修改的左值。
switch函数的优势与劣势
- 程序在switch中顺序执行,不受case标号影响。
switch (color) {
case 1:printf("red");
case 2:printf("yellow");
case 3:printf("blue");
}
若color=2
执行结果为 yellowblue
使用不对称边界
- 用第一个入节点和第一个出界点来表示一个数值的范围
一个字符串中由下标为16到下标为37的字符元素所组成的字串,它的长度是多少呢?
若将其表示为整数x满足x** >= 16(入界点)且x < **38(出界点),则很容易计算其结果。
该技巧可以用在for循环中,用来循环结束的标志。
使用头文件
- 每个外部对象只在一个头文件中声明,需要用到该外部对象的所有模块都包含这个头文件(包括其定义该外部对象的模块)
返回整数的getchar函数
- 单引号引起的字符代表一个整数。
- 若使用char接收,可能无法容下EOF。
宏可能产生的问题
- 宏定义中的空格
#define f (x) ((x)-1)
该宏定义为f代表(x) ((x)-1)
- 宏不是函数
#define abs(x) x>0?x:-x
abs(a-b)
展开后为
a-b>0?a-b:-a-b //并不是我们期望的-(a-b)
因此,宏定义中应该把每个参数都用括号括起来,整个表达式也应该用括号括起来。
- 存在副作用
#define max(a,b) ( (a) > (b) ? (a) : (b) )
biggest = max ( biggest , x[i++])
展开后
biggest = ( (biggest)>(x[i++]) ? (biggest) : (x[i++])
i一次可能递增2。
- 宏不是类型定义
#define T1 struct foo *
T1 a,b
展开为
struct foo * a , b
两者类型不同。