读书笔记--C陷阱与缺陷(二)

第二章

1. 理解函数声明

书中分析了复杂的类型声明方式,也说明了使用typedef声明会更好理解,推荐大家使用typedef进行函数声明。

书中类型分析一层一层挖掘,让读者可以理解多层嵌套的类型含义,有时间的读者可以去看看,笔者不再重复。

既然书中推荐使用typedef进行函数声明,我们就来研究下typedef:

typedef主要用于定义一种类型/结构体的别名。

从字面上看和C的宏定义 #define 挺像的,但是define只是简单的参数替换,如果不注意括号很容易产生预期之外的错误。

在指针类型声明时最好使用typedef,如下:

typedef char* PCHAR;

#define dpchar char*;

PCHAR p1,p2;

dpchar p3,p4;

上述语句中,p1,p2,p3可以被成功定义为char*,但是p4被定义为char(char* p3,p4;)。

对于结构体,一般定义为:

typedef struct STUDENT_ST

{

int id;

char name;

} student;

student st1;//代替struct STUDENT_ST st1,声明更加简单

typedef可高效解决平台无关的数据类型,这对公司非常重要,因为不同的项目背景可能使用的系统平台不尽相同,统一的管理数据类型可以避免错误。

如 typedef long double LONGNUM;

或 typedef long LONGNUM;

在跨平台时就可以通过修改typedef来切换数据类型,而不用修改源码,想象下如果修改每个源码里的类型该是多大的工程。。。

其实,还可以结合typedef使用bit field方式自定义字节长度的类型,下次有机会详细说明吧~~

typedef另一个使用较多的就是书中提到的声明复合类型:

如书中:(* (void (*) ()) 0) ();

可以改写为:typedef void (*pfun) ();

(* (pfun) 0) ();

2. 运算符优先级

其实记不清运算符优先级的时候怎么办,加括号呗,最多就是括号多了理解起来费时点~

来看看书中推荐的优先级记法:

第一优先级: ()  []  ->  .

其实是括号和调用操作符,很好理解,书中认为不用记,确实如此!

第二优先级: !  ~  ++  --  - (type)  *  &  sizeof

由于第一优先级好理解,单目运算符可认为是最高优先级。

但要注意两点:

  1. 对于 *p() ,因为第一优先级更高所以被解释为 *(p()),要正确调用指针p指向的函数应为 (*p)() 。
  2. 单目运算符是自右向左结合的(与人看文字方向相反)。所以如 *p++ 会被解释为*(p++),加1的是p值;写为 (*p)++,加1的是p指向对象的值;虽然都是取出p指向对象的值。。。。

第三优先级:  *  /  %  -  +  (算术运算符)

第四优先级: <<  >>  (移位运算符)

第五优先级: <  <=  >  >=  ==  !=  (关系/比较运算符)

第六优先级: &  |  ^  (逻辑/位运算符)

第七优先级: &&  ||  (逻辑运算符)

第八优先级: ? :  (条件运算符)

注意:关系/比较运算符中,==  != 的优先级低于其他关系运算符。如书中例子: a < b == c < d  即  (a<b) == (c<d)

逻辑运算符优先级也不相同,一般是”与”>”或”,即 &>^>|>&&>||

所有的赋值运算符(=)优先级一样,但是自右向左结合!!

3. 注意分号

主要注意if  while  struct声明。其实这些都比较简单,但是笔者确实遇到过这种问题,当时笔者在if();后加了分号,程序结果一直不对但是检查了很久才发现。。。

struct声明后紧跟函数声明时缺少分号很危险,书中提到编译器会把声明类型视为函数的返回值类型。如:

1 struct rec
2 {
3     int date;
4     int time;
5 }
6 main()
7 {
8     ...
9 }

struct rec的声明结尾没有分号,可能变为main函数的返回类型。。。

笔者对该程序进行了build,出现了两个warning和1个error:

第一个warning说:main函数的返回类型不是int;

第二个warning说:控制到达非void函数的结尾。通俗说就是应该带有返回值的函数结尾可能无返回值。

error说:类型不匹配,当要返回int却要求返回struct rec

看来优秀的IDE(codeblocks)还是可以避免这类错误的!

4. switch语句

书中强调C语言的switch语句能依次通过并执行各个case部分,所以若只想执行一个case要加break;

如:switch (color)

{

case 1 : printf(“red”);

case 2 : printf(“yellow”);

case 1 : printf(“blue”);

}

假设color取值2,最后程序打印  yellowblue

因为case2,case3都执行了,所以要加break;

当然如果你想通过刻意去掉break;来实现某些功能,可以这么写,但是代码阅读起来容易产生疑问。

时间: 2024-10-14 18:21:47

读书笔记--C陷阱与缺陷(二)的相关文章

读书笔记--C陷阱与缺陷(七)

第七章 1.null指针并不指向任何对象,所以只用于赋值和比较运算,其他使用目的都是非法的. 误用null指针的后果是未定义的,根据编译器各异. 有的编译器对内存位置0只读,有的可读写. 书中给出了一种判断编译器如何处理内存0的代码: 1 #include <stdio.h> 2 int main() 3 { 4 5 char *p; 6 p=NULL; 7 printf("location 0 contains: %d\n", *p); 8 9 return 0; 10

读书笔记--C陷阱与缺陷

要参与C语言项目,于是作者只好重拾C语言(之前都是C++,还是C++方便). 看到大家都推荐看看  C陷阱与缺陷(C traps and pitfalls),于是好奇的开始了这本书的读书之旅. 决定将书中重要的知识点和易错点记录下来方便自己复习和他人学习~~不多说了,下面开始. 第一章:词法陷阱 在C语言中,符号(程序文字)之间的空白(包括空格符.制表符.换行符)将被忽略.书中举了一例: 1 if (x > big) big = x; 2 可以写成: 3 if 4 ( 5 x 6 > 7 bi

读书笔记--C陷阱与缺陷(四)

第四章 1. 连接器 C语言的一个重要思想就是分别编译:若干个源程序可在不同的时候单独进行编译,恰当的时候整合到一起. 连接器一般与C编译器分离,其输入是一组目标模块(编译后的模块)和库文件,输出是一个载入模块(执行文件). 2. 命名冲突与static修饰符 static修饰符可有效减少命名冲突! 如: static int a; 与 int a; 声明含义相同,但是前者限制a的作用域在一个源文件(.c)内,其他源文件是不可见的.但后者都是可见的会产生命名冲突. 如果若干个函数需要共享一组外部

读书笔记--C陷阱与缺陷(六)

第六章 1.预处理器:预处理器先对代码进行必要的转换处理,简化编程者的工作. 它的重要原因有以下两点: a. 假如要将程序中出现的所有实例都加以修改,但希望只改动程序一处数值,重新编译实现. 预处理器可以做到这点,通过将这个数值设为显式常量. b. C语言函数调用花销大,希望有一个程序块看上去像函数确没有函数调用的开销. 2.宏定义的空格不能忽视 宏定义只是简单的文本替换,记住这个本质可以避免很多错误. 如: #define f  (x)  ((x)-1) 上式意思是: f(x)代表((x)-1

[读书笔记]了不起的node.js(二)

这周做项目做得比较散(应该说一直都是这样),总结就依据不同情境双开吧-这篇记录的是关于node的学习总结,而下一篇是做项目学到的web前端的知识. 1.HTTP篇 node的HTTP模块在第一篇时接触过,这里来学习几个例程中出现的API. 1 var qs = require('querystring'); 2 3 require('http').createServer(function(req, res){ 4 if('/' == req.url){ 5 res.writeHead(200,

c++ primer读书笔记之c++11(二)

1 新的STL模板类型,std::initializer_list<T> c++11添加了initializer_list模板类型,用于提供参数是同类型情况的可变长度的参数传递机制,头文件是<initializer_list>. 其具体接口可参考cplusplus.com的介绍,地址如下:http://www.cplusplus.com/reference/initializer_list/initializer_list/?kw=initializer_list 与vector不

(转)Spring读书笔记-----使用Spring容器(二)

一.使用ApplicationContext 前面介绍了,我们一般不会使用BeanFactory实例作为Spring容器,而是使用ApplicationContext实例作为容器,它增强了BeanFactory的功能. ApplicationContext允许以声明式方式操作容器,无须手动创建它.在Web应用启动时自动创建ApplicationContext.当然,也可以采用编程方式创建ApplicationContext. 除了提供BeanFactory所支持的全部功能外,Applicatio

《构建之法》读书笔记之:第一、二、十六章

这周看了邹欣老师<构建之法>的1,2,16章,获益匪浅.这本书写得妙趣横生,用阿超小飞几个人的生活场景和幽默的比喻帮我理解着软件工程的相关概念,让我对软件工程有了初步的了解:原来开发软件并不是我们想的那样简单:上手直接敲代码就可以了,而是会有一套详细的流程规范.下面是我看书时的一些心得笔记,和一些无法自己解答的疑惑,烦请各位老师批评指教. 第一章: 笔记: 软件=程序+软件工程,是否可以通俗地理解为,程序只是死的机器的东西,为什么做(需求分析),做什么(软件设计),做完后这东西是否可行(软件测

读书笔记 --TCP :传输控制协议(二)

TCP建立连接 请求端(客户端)发送一个SYN指明客户端打算连接的服务器端口号,以及初始序列号. 服务端发回包含服务器的初始序号的SYN报文段作为应答.同时,将确认序号设置为客户的ISN加1以对客户的SYN报文段进行确认.一个SYN占用一个序号. 客户端必须将确认序号设置为服务端的ISN加1以对服务器的SYN报文段进行确认. TCP连接终止 建立一个连接需要三次握手,而终止一个连接要经过4次握手.这是由TCP的半关闭造成的.既然一个TCP连接是全双工(即数据在两个方向上能同时传递),因此每个方向