c 陷阱与缺陷(一)】的更多相关文章

要参与C语言项目,于是作者只好重拾C语言(之前都是C++,还是C++方便). 看到大家都推荐看看  C陷阱与缺陷(C traps and pitfalls),于是好奇的开始了这本书的读书之旅. 决定将书中重要的知识点和易错点记录下来方便自己复习和他人学习~~不多说了,下面开始. 第一章:词法陷阱 在C语言中,符号(程序文字)之间的空白(包括空格符.制表符.换行符)将被忽略.书中举了一例: if (x > big) big = x; 可以写成: if ( x > big ) big = x ;…
这段时间把<C陷阱和缺陷>看了,没时间自己写总结.就转一下别人的学习笔记吧http://bbs.chinaunix.net/thread-749888-1-1.html Chapter 1 词法陷阱 程序中的单个字符孤立起来看并没有什么意义,只有结合上下文才有意义,如p->s = "->";两处的-意义是不同的. 程序的基本单元是token ,相当于自然语言中的单词. 一个token的意义是不会变的. 而组成token 的字符序列则随上下文的不同而改变. tok…
第七章 1.null指针并不指向任何对象,所以只用于赋值和比较运算,其他使用目的都是非法的. 误用null指针的后果是未定义的,根据编译器各异. 有的编译器对内存位置0只读,有的可读写. 书中给出了一种判断编译器如何处理内存0的代码: #include <stdio.h> int main() { char *p; p=NULL; printf("location 0 contains: %d\n", *p); ; } 在禁止读内存0的机器上,程序会执行失败.有的机器上,却…
版权声明:本文为Focustc原创文章.转载请注明作者及出处. https://blog.csdn.net/caozhankui/article/details/35925939 看完<C陷阱与缺陷>,忍不住要又一次翻一下,记录一下与自己的惯性思维不符合的地方. 记录的是知识的增量.是这几天的流量,而不是存量. 这本书是在ASCI C/C89订制之前写的.有些地方有疏漏. 第一章 词法陷阱 1.3 C语言中解析符号时使用贪心策略,如x+++++y将被解析为x++ ++ +y.并编译出错. 1.…
有感自己的C语言在有些地方存在误区,所以重新仔细把"C陷阱和缺陷"翻出来看看,并写下这篇博客,用于读书总结以及日后方便自身复习. 第1章 词法"陷阱" 1.1 =不同与==   = 是赋值操作符,而 == 是作为比较操作符,初学者容易将 == 错写为 = ,这种情况下编译器不会报错,这就有可能造成很严重的后果,还不容易发现.比如下面这个例子: while( c=' ' || c=='\t' || c=='\n' ) { ; }   即使c既不等于'\t',也不等于'…
下载地址:点我 凯尼格 (作者), 高巍 (译者) <C和C++经典著作:C陷阱与缺陷>适合有一定经验的C程序员阅读学习,即便你是C编程高手,<C和C++经典著作:C陷阱与缺陷>也应该成为你的案头必 备书籍.作者以自己1985年在Bell实验室时发表的一篇论文为基础,结合自己的工作经验扩展成为这本对C程序员具有珍贵价值的经典著作.写作<C和C++经典著作:C陷阱与缺陷>的出发点不是要批判C语言,而是要帮助C程序员绕过编程过程中的陷阱和障碍.<C和C++经典著作:C…
1. 词法“陷阱” = 不同于 == , 可以通过if( 1 == a )来避免 & | 不同于 && || 词法分析中的“贪心法” 编译器将程序分解成符号的方法是,从左到右一个字符一个字符地读入,如果该字符可能组成一个符号,那么再读入下一个字符,判断已经读入的两个字符组成的字符串是否可能是一个符号的组成部分:如果可能,继续读入下一个字符,重复上述判断,直到读入的字符组成的字符串已不再可能组成一个有意义的符号.这个处理策略被称为“贪心法”,也称为“大嘴法”.举例:a---b 等价于…
本文关注的主要是python陷阱,具体而言,是指CPython,而且除非特别说明,所有代码示例都是在python2.7运行的. 我个人对陷阱的定义是这样的:代码看起来可以工作,但不是以你“想当然“”的方式.如果一段代码直接出错,抛出了异常,我不认为这是陷阱.比如,Python程序员应该都遇到过的“UnboundLocalError", 示例: >>> a=1 >>> def func(): ...     a+=1 ...     print a ... &g…
笔记: 第一章:词法陷阱 提倡显示比较if((x = y) != 0) foo(); 第二章:语法陷阱 已知一个类型的声明 该类型的类型转换:吧声明中的变量名和声明末尾的分号去掉,再将剩余的部分用括号整个“封装”起来即可 (*(void(*)())0); (void(*)()) 这是一个返回值为void的函数指针类型 typedef void (*funcptr)() 中 funcptr 等价于 (void(*)()) typedef void (*funcptr)(); //funcptr表示…
第一章 词法“陷阱” 1. =不同于== if(x = y) break; 实际上是将y赋给x,再检查x是否为0. 如果真的是这样预期,那么应该改为: if((x = y) != 0) break; 2. &和| 不同于 && 和 ||   3.词法分析中的“贪心法” 编译器将程序分解成符号的方法是:从左到有一个一个字符的读入,如果该字符可能组成一个符号,那么再读入下一个字符,判断已经读入的两个字符组成的字符床是否可能是一个符号的组成部分:如果可能,继续读入下一个字符,重复上述判断…
第3章 语义"陷阱"     一个句子哪怕其中的每个单词都拼写正确,而且语法也无懈可击,仍然可能有歧义或者并非书写者希望表达的意思.程序也有可能表面上是一个意思,而实际上的意思却相去甚远,本章考查了若干种可能引起上述歧义的程序书写方式. 3.1 指针与数组     C语言中指针与数组这两个概念之间的联系是如此密不可分,以至于如果不能理解其中一个概念,就不可能理解另一个概念.而且,C语言对这些概念的处理, 在某些方面与其它任何为人熟知的程序语言都不同.   C语言的数组值得注意的地方有以…
第一章 词法"陷阱" 1.4整型常量 如果一个整型常量的第一个字符是数字0,那么该常量将被视作八进制数.因此,10与010的含义截然不同.需要注意这种情况,有时候在上下文为了格式>对齐的需要,可能无意中将十进制数写成了八进制数,例如: struct{ int part_number; char *description; }parttab[] = { 046, "left-handen widget" , 047, "right-handed wid…
词法分析: 编译器将程序分解成符号的方法是 从左到右一个一个字符的读入,如果该字符可能组成一个符号,再读入下一个字符 而c语言里的符号 / * =只有一个字符长, 是单字符的, /* == 一些事双字符符号. 故因此有以下陷阱 a---b     等价     a-- -b y= x/*p      /*被编译器理解成一段注释的开始 因此应该写成y = x / *p 或者 y = x/(*p) 整型常量 : 如果一个整形常量的第一个字符是数字0 那么该常量将被视作八进制,有时候在上下文中为了格式…
第三章 语义陷阱 3.1 指针与数组 C语言中只有一维数组,而且数组的大小必须字编译期就作为一个常数确定下来.数组中的元素可以是另外一个数组.任何一个数组下标运算都等同于一个对应的指针运算.int a[3];除了a被用作运算符sizeof的参数这一情形,在其它所有情形中数组名a都代表指向数组a中下标为0的元素的指针,sizeof(a)是整个数组a的大小.*a是数组a中下标为0的元素的引用,*(a+1)是数组中下标为1的元素的引用. 3.2 非数组的指针 在C语言中,字符串常量代表了一块包括字符串…
第二章 语法陷阱 2.1 理解函数声明 (*(void(*)())0)();任何C变量的声明都由两部分组成:类型以及一组类似表达式的声明符.一旦我们知道了如何声明一个给定类型的变量,那么该类型的类型转换符只需要添加一个括号即可的到.例如 float (*h)();表示h是一个指向返回值为浮点类型的函数的指针.因此(float (*h)())表示一个"指向返回值为浮点类型的函数的指针"的类型转换符. 第一步假设fp是一个函数指针,那么通过(*fp)()调用fp所指向的函数,可以简写问fp…
第一章 词法陷阱 术语“符号”指的是程序的一个基本组成单元,其作用相当于一个句子中的单词.编译器中负责将程序分解为一个一个符号的部分,一般称为“词法分析器”. 1.1 =不同于== 一般容易将比较运算误写成赋值运算的情形,另一方面,如果把赋值运算误写成比较运算,同样会造成混淆:if((filedesc==open(argv[i],0))<0)error(); 在本例中,如果函数open执行成功,则返回0或者正数:而如果函数open执行失败,将返回-1.上面这段代码的本意是将函数open的返回值存…
作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz 陷阱1 理解函数声明 作者提出一个问题:有一个首地址为0的函数,该函数返回值类型为void,没有参数.怎样用C语言的语句调用这个函数? 答案是(*(void  (*)())0)(); 要理解这个调用形式,要清楚如下两个问题: 一是函数指针. 假设fp是一个函数指针,则调用fp所指向的函数的方法是 (*fp)(); 因为fp是一个函数指针,所以*fp是该指针所指向的函数,所以(*fp)()就是调用该函数的方式.AN…
作者:刘昊昱 博客:http://blog.csdn.net/liuhaoyutz 编译器中负责将程序分解为一个一个符号的部分,称为“词法分析器”.下面看一个例子: if(x > big) big = x; 这个语句的第一个符号是C语言的关键字if,紧接着下一个符号是左括号,再下一个符号是标识符x,再下一个是大于号,再下一个是标识符big,依次类推.在C语言中,符号之间的空白符将被忽略. 本章将探讨符号和组成符号的字符间的关系,以及有关符号含义的一些常见误解. 陷阱1 “=”不同于“==” 将相…
前言和导读 "得心应手的工具在初学时的困难程度往往超过那些easy上手的工具."比較认同这句话. 我至今认为自己事实上还是个刚入了门的刚開始学习的人. 第一章 "词法"陷阱 因为之前学过编译原理,对编译器词法分析(主要是符号识别过程)比較了解,理解起来不困难. 在讲到"="和"=="."|"和"||"."&"和"&&"时,联…
第2章 语法陷阱 2.1 理解函数声明   当计算机启动时,硬件将调用首地址为0位置的子例程,为了模拟开机时的情形,必须设计出一个C语言,以显示调用该子例程,经过一段时间的思考,得出语句如下: ( *(void(*) () )0 ) ();   像这样的表达式看起来很难理解,但只要将其一层一层地剥离,还是能够理解的.下面我将用几个例子来帮助大家逐渐理解这个表达式. void *a(); void (*b) ();   因为()的优先级高于,所以a()为(a()),a是一个函数,该函数的返回类型为…
本文转自 https://www.xuebuyuan.com/1951579.html 自己找工作过程中复习过的书包括<C traps and Pitfalls>,<编程珠玑>,<编程之美>,<算法导论>,<Effective c++>,<c++ 沉思录>,另外的就是Linux我平时经常看的一些书了.所以这些书都是在我的书桌上,从来没有丢过.当然还有一本你永远都不可能撤下去的书叫做<深入理解计算机系统>,这个自然是不用提的…
Objective-C是一个强大而且非常有用的语言,但是同样也是有一点危险的.这次主题是受到一篇有关C++陷阱的文章启发,来聊聊Objective-C和Cocoa中的陷阱. 简介 我将和Horstmann使用同样的定义:陷阱是能够编译.链接.运行,但却不会按你所预期地去执行的代码.他提供了一个例子,这段代码在Objective-C中和在C++中同样都是有问题的: <span style=;</span> 肤浅地阅读这段代码可能会认为,它用来检查x是不是在[-0.5,0.5]区间内.但并不…
导读 程序是由符号(token)序列所组成的,将程序分解成符号的过程,成为"词法分析". 符号构成更大的单元--语句和声明,语法细节最终决定了语义. 词法陷阱 符号(token)指的是程序的一个基本组成单元,其作用相当于一个句子中的单词. 编译器中负责将程序分解为一个一个符号的部分,称作"词法分析器". 在C语言中,符号之间的空白(/b /t /n...)将被忽略. #include <stdio.h> int main() { if ( 1 ) pri…
编译器中负责将程序分解为一个一个符号的部分,一般称为"词法分析器".在C语言中,符号之间的空白(包括空格符.制表符或换行符)将被忽略. 1.=不同于== C语言使用符号"="作为赋值运算符,符号"=="作为比较.赋值运算相对比较运算出现得更频繁,因此字符较少的符号"="就被赋予了更常用的含义--赋值操作.C语言中赋值符号被作为一种操作符对待,因而重复进行赋值操作可以很容易地书写,并且赋值操作还可以被嵌入到更大的表达式中. 这种…
1,数组 对数组只能进行两种操作,1确定数组的大小,2获得数组第一个元素的指针,其他的操作均是通过指针来实现的. 1 2 3 4 5 6 7 8 9 #include <stdio.h>   main() {     int a[2][3]={{1,2,3},{4,5,6}};     int *p,(*q)[3];     p=a[2];     q=a; } int a[2][3]={{1,2,3},{4,5,6}}; a为一个数组(A),数组的维度为3,元素为数组(B),数组(B)的维度…
1,符号之间的空白被忽略 符号中间不能嵌入空白 2,词法分析中的贪心法 a---b 和 a-- -b相同 和 a- --b不同 1 a=b/*p //根据贪心法 /*被解释成 注释符,便不再往下读,直到匹配 */ 1 a=b/ *p//*号优先级大于 / 故为b除以 p所指向的值 根据编程规范,方便阅读,减少出错的可能 1 a=b/(*p) 3,理解函数的声明 4,运算符的优先级 5,分号的使用 1 1 2 if() a; 1 2 if(); a; 2 1 2 3 return; a; b; 1…
宏: 宏不是函数 展开会产生庞大的表达式 #define MIN(A,B) ((A) <= (B) ? (A) : (B))MIN(*p++, b)会产生宏的副作用 剖析: 这个面试题主要考查面试者对宏定义的使用,宏定义可以实现类似于函数的功能,但是它终归不是函数,而宏定义中括弧中的“参数”也不是真的参数,在宏展开的时候对“参数”进行的是一对一的替换.程序员对宏定义的使用要非常小心,特别要注意两个问题: (1) 谨慎地将宏定义中的“参数”和整个宏用用括弧括起来.所以,严格地讲,下述解答: #de…
1.程序在设计时,往往得出正确的结果,但是它并不是程序员自己想要的. 例如: printf("hello world!") 编译器进行编译时不会出现任何问题,但是结果: 提示竟然出现在!的后面.因此   \ n   的使用变的尤为重要. 2. = 和 == while( c =  '  ' ||  c == '  \t  '   ||  c ==  ' \  n' ) 这段程序程序员想表达的意思 当 c 为 空格.制表符.换行时,执行子程序.但是如果将== 写成 = 时,由于赋值运算符…
第四章 连接 4.1 什么是连接器 C语言中的一个重要思想就是分别编译,即若干个源程序可以在不同的时候单独进行编译,然后在恰当的时候整合在一起.典型的连接器把由编译器或汇编器生成的若干个目标模块,整合成一个被称为载入模块或可执行文件的实体,该实体能够被操作系统直接执行.连接器通常把目标模块看成是由一组外部对象组成的.连接器的输入一组目标模块和库文件,连接器的输出是一个载入模块. 4.2 声明与定义 下面的声明语句 int a;如果其位置出现在所有的函数体之外,那么它就被称为外部对象a的定义.这个…
1.数组名作实參     在C语言中,我们没有办法将一个数组作为函数參数传递,假设我们使用数组名作为參数.这个时候数组名立马会被转换为指向该数组的第一个元素的指针.     关于这一点的理解能够向前深入一步,比方定义的数组为int a[3],那么a作为參数传递之后会变为int *类型:假设定义的数组为int a[3][4],那么a作为參数传递之后被变为int (*)[4].假设定义的数组为int a[3][4][5].那么a作为參数传递之后会变为int (*)[4][5]:兴许的以此类推.为什么…