
C语言会同意一些"令人震惊"的结构,下面的结构是合法的吗,我们来看看几个例子。 c = a+++b;以下代码是合法的吗,咋的一看不禁有这样的疑问? int a = 5, b = 7, c;c = a+++b; 这个代码确实不咋符合习惯的写法,但是不管你相不相信,上面的例子是完全合乎语法的。问题是编译器如何处理它? 根据最处理原则,编译器应该能够尽可能处理所有合法的用法。因此,上面的代码会被处理成: c = a++ + b;我们来测试一下: //https://tool.lu/coderunner/ o0 }8 f s& \6 G+ Q6 n0 J: h//来源:技术让梦想更伟大 //作者:李肖遥 3 G8 `# r; ~6 @8 y- h #include <stdio.h> int main() [8 I0 h3 z+ n/ f6 ~: O" ~ { int a = 5, b = 7, c; c = a+++b;3 M) ^% m7 \7 S+ B printf("a = %d,b = %d,c = %d",a,b,c); return 0; } 输出结果如下:
其执行顺序:
上面那么可能只是开胃菜,那么这个表达式呢,有什么想法?咱们二话不说,直接上代码测试。 //https://tool.lu/coderunner/& k* Q$ N3 |7 t# @ f$ Y//来源:技术让梦想更伟大+ N& d+ j5 T @3 b6 s* z3 G //作者:李肖遥 6 _, j: m% s8 b #include <stdio.h> int main()9 _6 U, B/ o) d1 _ {4 W' V" h6 J1 P ?1 s6 g2 b5 b1 E int a = 5, b = 7, c;: L( U7 Y+ \' U% B9 K+ j c = a+++++b;# }: S1 d: _6 b) j7 m) a5 m- | //c = (a++) + (++b);( n( ~' n" J6 r4 q# b7 s3 a //c = ((a++)++) + b; printf("a = %d,b = %d,c = %d",a,b,c); return 0; }9 W+ p7 K) |: z/ h5 e! w 编译结果如下:
虽然一看就知道意思是(a++)+(++b);,但是编译就通不过,我们把括号加上,c = (a++) + (++b);,编译通过,那么出错信息大概是:括号影响了优先级?我们从侧面以及原理来解析一下。 侧面解析一下对于a+++++b这一段代码,编译系统从左至右扫描整条语句,先遇到a++,判断出来是一个a的后缀自加运算; 然后接着扫描,遇到一个+,+是一个二目运算符,它的左边已经有一个运算数a++了,系统就向右搜索第二个运算数; 又遇到一个+,++比+的运算级别要高,这时,编译系统就将两个+看成一个整体来处理; 既然是++,编译系统就认定,肯定它的左边或右边有一个变量,编译系统先搜索左边,发现++,不是变量; 再搜索右边,发现+b,+b是什么东西?编译系统是无法搞明白的; 因此它就认为++是一个缺少左值的自增运算符,于是提示错误给用户:lvalue required as increment operand 原理解析一下C语言在这里遵循词法解析的贪婪匹配原则。优先匹配尽可能多字符的符号,无论是否有语法错误(因为词法分析时还没有语法检查)。 于是a+++++b会被当作a ++ ++ + b,这是非法的表达式,因此产生编译错误。 这个问题在ISO C99标准中直接以示例描述,原文如下: “那么,也许是c = ((a++)++) + b;这个意思呢? 我们编译结果依然报错,如下:
这涉及到“左值”的问题。百度百科定义如下: “ 总结:a存在左值,可以有表达式:a++。但是(a++)不存在左值,无法继续执行(a++)++操作,所以最终报错。 怎么样才能编译正确呢?a+++++b在编译时会报错,那么a++ + ++b呢?来看以下代码: //https://tool.lu/coderunner///来源:技术让梦想更伟大 `7 s& z0 G# S5 C* ~4 L# c0 G' s //作者:李肖遥 L# i3 w0 ]! t. N& J #include <stdio.h> int main()( p* E5 B4 z) X { int a = 5, b = 7, c;( }7 ^6 H/ P c: a" ]2 j R c = a++ + ++b; printf("a = %d,b = %d,c = %d",a,b,c); return 0; }7 v& |; U- R$ @4 g3 j- o 结果如下:
其代码与c = (a++) + (++b);结果一样,说明是正确的,其按照下面顺序执行:
这个表达式看着爽不爽?我们知道自增自减运算,表示对自身进行a=a+1或者a=a-1的运算。 ++a表示在调用前就a+1,a++表示在调用后+1。 int c=(++a,b++,a++,++b);这个逗号隔开的表示用最后一个式子对C进行赋值,测试如下: //https://tool.lu/coderunner///来源:技术让梦想更伟大 //作者:李肖遥 : L- S2 l$ { f E4 ~$ k #include <stdio.h>4 Z6 e' z+ j7 d; ~9 I7 Y' S R 7 H- I! U$ i3 }0 n( [8 o6 R) u int main()* j; Y z, d/ ]# u% u3 j { int a = 5, b = 7, c;% c: H7 ^; i8 D c=(++a,b++,a++,++b); printf("a = %d,b = %d,c = %d",a,b,c);7 K6 d$ J: d! U3 x' d7 S* s; k return 0;, t @( `# x [+ F0 F. H- W } 输出的结果如下:
这段执行的顺序如下
我们看到这些有趣且不常见的代码的时候,无需怀疑,只需要根据经验一步一步分析,就会发现其中的玄妙之处。在此我总结以下几点。
|
我从来不这么干!不给自己个别人找麻烦! |
这种装模作样的代码语法就国人喜欢拿来装逼用而已,代码是拿来阅读改进的,浪费时间去增加阅读难度还不如好好想想你的应用逻辑怎么改进优化![]() |
【STM32N6570-DK评测】摄像头video encoder
实战经验 | STM32CubeIDE实用技巧之工程联调
【STM32H745I-DISCO】基于TouchGFX的工业控制器界面设计
【下载问题解决】关于ST官网下载软件问题解决
【STM32N6570-DK评测】开发环境及LED debug
实战经验 | 基于STM32CubeIDE下载TouchGFX GUI应用的出错分析
实战经验 | 如何修改STM32Cube固件包的存储位置
兔哥的杂谈【002】——如何性价比更高地去编译STM32
IDE删除编译中间文件
STM32CubeIDE 快速入门指南