C陷阱与缺陷之语法陷阱】的更多相关文章

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