《C陷阱与缺陷》读书笔记

词法分析中的“贪心法”
  • 每个符号应该包含尽可能多的字符。
    a---b等同于(a--)-b
    a+++++b被编译器视为((a++)++)+b,语法上是不正确的。
字符与字符串
  • 单引号引起的字符代表一个整数。

  • 双引号引起的字符串代表一个指向无名数组起始字符的指针。

理解函数声明
(* (void (*) () ) 0) ();

这是一个经典的例子。分析如下:

  1. void (*) () 指向返回值为void类型的函数指针

  2. ( void (*) () ) 0 将0进行类型转换

  3. (* ( 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
宏可能产生的问题
  1. 宏定义中的空格
#define f (x) ((x)-1)

该宏定义为f代表(x) ((x)-1)

  1. 宏不是函数
#define abs(x) x>0?x:-x
abs(a-b)

展开后为

a-b>0?a-b:-a-b //并不是我们期望的-(a-b)

因此,宏定义中应该把每个参数都用括号括起来,整个表达式也应该用括号括起来。

  1. 存在副作用
#define max(a,b) ( (a) > (b) ? (a) : (b) )
biggest = max ( biggest , x[i++])

展开后

biggest = ( (biggest)>(x[i++]) ? (biggest) : (x[i++])

i一次可能递增2。

  1. 宏不是类型定义
#define T1 struct foo *
T1 a,b

展开为

struct foo * a , b 

两者类型不同。

你可能感兴趣的:(《C陷阱与缺陷》读书笔记)