笔记


第一章:词法陷阱

提倡显示比较
if((x = y) != 0)
foo();

第二章:语法陷阱

已知一个类型的声明
该类型的类型转换:吧声明中的变量名和声明末尾的分号去掉,再将剩余的部分用括号整个“封装”起来即可
(*(void(*)())0);
(void(*)()) 这是一个返回值为void的函数指针类型
typedef void (*funcptr)() 中 funcptr 等价于 (void(*)())

typedef void (*funcptr)(); //funcptr表示返回值为void的函数指针,
可以引用同类型返回值的函数,有无参数都可以,
但是如果显式声明带有参数,如typedef void (*funcptr)(int);则只能引用带有一个int参数返回void的函数
当然没有参数的函数可以传递一个无效的参数进行引用
void test(){return;} funcptr cp = (funcptr)test; cp(0);

优先级问题:
单目 大 双目 (移位 小 算术 关系紧跟后) 逻辑 赋值 条件
终极必杀: 加括号

第三章:语义陷阱

C语言中只有一维数组,而且数组的大小必须在编译期就作为一个常数确定下来
二维数组就是以数组为元素的数组
int (*ap)[31] : *ap是一个拥有31个整型元素的数组,因此ap就是一个指向这样的数组的指针

空指针 p = NULL 地址是无效的 并不是空字符串 空字符串是地址是有效的 只不过内存中的内容为空

声明缓冲区
#define N 1024
static char buffer[N];

提到一点:--n 可能要比 n++ 快 理由:n++要先保存n在减1 (理由不是太懂,感觉写的不清楚)

运算符 && || 首先对左侧操作数求值,只在需要的时候才对右侧操作数求值 -> 让最可能对表达式起到作用的操作数在左边(提高速度)

对于数组结尾之后的下一个元素,取它的地址是合法的,但是取地址内的内容是不合法的

整数溢出
无符号算术运算是以2的n次方为模(n是结果中的位数),相当于时钟循环 11点加2个小时 是1点(实际上也溢出了)
两个操作数都是有符号整数时,“溢出”会出现
if((unsigned)a + (unsigned)b > INT_MAX):转换为无符号数来判断是否溢出
char a = 126;
char b = 125;
printf("十六进制:%x\n",a);
printf("十六进制:%x\n",b);
printf("十进制:%d\n",a);
printf("十进制:%d\n",b);
printf("%d\n", CHAR_MAX);
printf("-------------------\n");
printf("十六进制:%x\n", (unsigned char)a + (unsigned char)b);
printf("十进制:%d\n", (unsigned)a + (unsigned)b);
printf("差:%d\n", (unsigned)a + (unsigned)b - CHAR_MAX);

第四章:连接

若干个文件分别编译 连接器将若干个C源程序整合成一个整体

外部变量存在定义与声明 声明可以有多个 定义只能有一个 如果没有赋初始值 会自动初始化为0
前面加 static修饰 可以起到“屏蔽”的作用 使得被修饰的变量或函数 值在本源文件中可见

定义:char filename[] = "/etc/passed"; /*filename是一个字符数组的名称 类型是“字符数组”*/
另一文件中声明:extern char *filename; /*filename是一个字符指针 类型是指针*/

C语言的规则:如果一个未声明的标识符后跟一个开括号 那么它将被视为一个返回整型的函数
main(){} 有返回值并且是整型

第五章:库函数

getchar函数的返回值类型是 整数(int)为了可以接收getchar返回的全部 需要定义一个 int类型的变量来接收它的返回值

更新顺序文件
为了保持与过去不能同时进行读写操作的程序的向下兼容性,一个输入操作不能随后直接紧跟一个输出操作,反之亦然。
如果要同时进行输入和输出操作,必须在其中插入fseek函数的调用

缓冲区输出与内存分配:setbuf(stdout, buf) 的使用有风险 ---- 经典错误

errno signal 的使用复杂

第六章:预处理器

特别注意 宏定义中的空格 #define f (x) ((x) - 1) f -> (x) ((x) - 1)

宏不是函数 只是一种替换 要多加括号防止意外
带有多次计算的时候 要特别注意
宏不是语句 定义时候一般不需要加 分号
宏不是类型定义
#define INTP int *
INTP a, b; 替换后 int * a, b; 则a是int指针 b是int数据
typedef int *INTP;
INTP a, b; a b都是int指针

第七章:可移植性缺陷

函数声明中略去参数类型的说明,这在ANSI C 标准中也是合法的
因为这样的声明并没有对参数类型做出任何说明,就意味着如果在函数调用时传入了错误类型的参数,函数调用就会不声不响地失败

C语言实现必须能够区别出前6个字符不同的外部名称,而且,这个定义中并没有区分大小写字母

为了移植性可以定义自己的数据类型:typedef long my_long

(unsigned) c :c从char型转换为int型整数 再转换为无符号int 从8位到32位
(unsigned char) c :c从char转换为无符号char
char c = 'a';
printf("%d\n", sizeof((unsigned char) c)); //输出1
printf("%d\n", sizeof((unsigned) c)); //输出4

移位运算
左移 后面补0
右移 逻辑 :无符号 使用逻辑:左边补0 / 算术 :有符号 使用算术:左边补符号位的副本
一般非负 右移一位 <=> 除2 (整数运算)

先释放在重新分配 --- 按照自己现有的方案写

字符串常量可以用来表示一个字符数组
"0123456789"[2] => '2' //取数组的下标为2的字符

建议

不要舒服自己相信“皇帝的新装”
while(c == 't' || c = ' ' || c == '\n')
c = getc(f);

直接了当地表明意图
当你编写代码的本意是希望表达某个意思,但这些代码有可能被误解为另一种意思时,请用括号或者其他方式让你的意图尽可能清楚明了
while('\t' == c || ' ' == c || '\n' == c)
c = getc(f);

考察最简单的特例
着重注意输入数据为空或者只有一个元素

使用不对称边界
注意数组小标从0开始

注意潜伏在暗处的Bug
程序的生命期往往要长于它运行其上的机器的生命期(可移植性)

防御性编程
多考虑坏的情况并加以处理

附录A

精度修饰符:指定打印数字的最小位数,不足前面补0
printf("%.3d\n", 7); 007 //整数---一共的位数
printf("%03d\n", 7); 007
printf("%.3f\n", 7.2345); 7.235 //小数---小数点后的位数

修饰符 l(long) 只对用于整数的格式码有意义
但是printf("%lf\n", 3.14); 可以打印出 3.140000

可变参数...


本节完......

《C陷阱与缺陷》阅读笔记(个人版)的更多相关文章

  1. C陷阱和缺陷学习笔记

    这段时间把<C陷阱和缺陷>看了,没时间自己写总结.就转一下别人的学习笔记吧http://bbs.chinaunix.net/thread-749888-1-1.html Chapter 1 ...

  2. C的陷阱和缺陷研读笔记01

    词法分析: 编译器将程序分解成符号的方法是 从左到右一个一个字符的读入,如果该字符可能组成一个符号,再读入下一个字符 而c语言里的符号 / * =只有一个字符长, 是单字符的, /* == 一些事双字 ...

  3. C陷阱与缺陷学习笔记

    导读 程序是由符号(token)序列所组成的,将程序分解成符号的过程,成为"词法分析". 符号构成更大的单元--语句和声明,语法细节最终决定了语义. 词法陷阱 符号(token)指 ...

  4. C的陷阱和缺陷研读笔记02

    宏: 宏不是函数 展开会产生庞大的表达式 #define MIN(A,B) ((A) <= (B) ? (A) : (B))MIN(*p++, b)会产生宏的副作用 剖析: 这个面试题主要考查面 ...

  5. 《c陷阱与缺陷》笔记--注意边界值

    如果要自己实现一个获取绝对值的函数,应该都没有问题,我这边也自己写了一个: void myabs(int i){ if(i>=0){ printf("%d\n",i); }e ...

  6. 《c陷阱与缺陷》笔记--移位运算

    #include <stdio.h> int main(void){ int a = 2; a >> 32; a >> -1; a << 32; a & ...

  7. C陷阱与缺陷读书笔记

    2.1理解函数声明 这一章仔细分析了(*(void(*)())0)();这条语句的含义,并且提到了typedef的一种函数指针类型定义的用法. 我们经常用到的typedef用法是用于指定结构体的类型, ...

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

    第七章 1.null指针并不指向任何对象,所以只用于赋值和比较运算,其他使用目的都是非法的. 误用null指针的后果是未定义的,根据编译器各异. 有的编译器对内存位置0只读,有的可读写. 书中给出了一 ...

  9. 读书笔记--C陷阱与缺陷(一)

    要参与C语言项目,于是作者只好重拾C语言(之前都是C++,还是C++方便). 看到大家都推荐看看  C陷阱与缺陷(C traps and pitfalls),于是好奇的开始了这本书的读书之旅. 决定将 ...

  10. 阅读《C陷阱与缺陷》的知识增量

    版权声明:本文为Focustc原创文章.转载请注明作者及出处. https://blog.csdn.net/caozhankui/article/details/35925939 看完<C陷阱与 ...

随机推荐

  1. SQL Server - 开窗函数

    -- 开窗函数:在结果集的基础上进一步处理(聚合操作) SELECT * FROM dbo.Student S# Sname Sage Ssex ---------- ---------- ----- ...

  2. [Java] Servlet工作原理之二:Session与Cookie

    (未完成) 一.Cookie与Session的使用简介 1 Cookie Cookie 用于记录用户在一段时间内的行为,它有两个版本:Version 0 和 Version 1,分别对应两种响应头 S ...

  3. OpenStack中MySQL高可用配置

    采用Heartbeat+DRBD+mysql高可用方案,配置两个节点的高可用集群 l  配置各节点互相解析 gb07 gb06 l  配置各节点时间同步 gb07 [root@gb07 ~]# ntp ...

  4. Google Email 帐户泄露

    最初爆出来的网站是:https://forum.btcsec.com/index.php?/topic/9426-gmail-meniai-parol/,是一个俄罗斯论坛,然后..就流传开来了... ...

  5. spring中set注入的一些小细节错误

    这是小白偶尔一直null指针的错误,调试了好久,原来是自己对spring注入的不够了解 我相信有很多跟我差不多的初学者会遇上,所以特地写出来,防止有人跟我一样.哈哈,也写上去,以防自己下次还犯这样的错 ...

  6. 一些CSS3的乐趣 - 工作也能发现乐的源头

    中秋节 translate 前些日子做一个中秋节的专题,主要就是写一个效果,月亮滚动,花瓣飘落.具体代码如下: .icons {z-index:10088; position:absolute; -w ...

  7. 20155212 2016-2017-2 《Java程序设计》第6周学习总结

    20155212 2016-2017-2 <Java程序设计>第6周学习总结 教材学习内容总结 Chapter10 输入串流为java.io.InputStream,输出串流为java.i ...

  8. Linux下可以使用ps命令来查看Oracle相关的进程

    Linux下可以使用ps命令来查看Oracle相关的进程 Oracle Listener 这个命令会列出Oracle Net Listener的进程 [oracle@ www.linuxidc.com ...

  9. Number of Islands I & II

    Given a boolean 2D matrix, find the number of islands. Notice 0 is represented as the sea, 1 is repr ...

  10. 总结WCF开发中遇到的几个问题

    最近的项目,需要用到WCF,在以前的工作中,经常是将WCF托管在IIS中,主要有几下几个原因:      第一:部署非常方便,和部署一个站点没什么区别:      第二:不受防火墙的影响,因为一般服务 ...