|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
" D, y3 C4 H7 P$ y3 R i5 m$ w
gcc对C语言的扩展:语句内嵌表达式(statement-embedded expression)
. H% i% J: G; n" m0 I! p; j- \& L今天突然发现了这个新的语法现象,现转载如下。感觉到目前为止,内嵌表达式有点像“鸡肋”,不好用但又确实有其好处。不过还是不建议使用,现转载的目是希望以后大家在看系统代码时遇到这种现象能知其所以原,出现BUG也好应对。5 r" q1 {6 R$ o4 G' E/ }
% s: b2 g# F% a/ u0 X* H: z, \. C
% J' y- }% G( L% s% m6 p在GNU C 中,用括号将复合语句括起来也形成了表达式。他允许你在一个表达式内使用循环,跳转和局部变量。
& `& K* V% ^( O3 W! Q# C7 F# w( C0 s7 b5 w
一个复合语句是用大括号{}括起来的一组语句。在包含语句的表达式这种结构中,再用括号( )将大括号括起来,例如:
$ Z+ @. b1 o# M/ g1 r/ p& x({ int y = foo (); int z;
: s: [0 V% y- J3 K8 a if (y > 0) z = y;
. ~& ?) M4 T2 Y( G4 B# e* x else z = - y;
6 ?+ N3 @, c7 D& ?1 J0 |. n z; })1 d! O1 R- S3 n& u$ Z% o
% S( f- [7 w$ a# z9 L/ R就是一个合法表达式,用于计算foo( )函数返回值的绝对值。
7 c/ V# B0 f% `0 [! c% e, p在上面的复合语句中,最后的一句必须是一个以分号结尾的表达式。这个表达式代表了整个结构的值。如果你在大括号里的最后一句用的是其他的语句,则整个结构的返回类型为void,即没有合法的返回值。' g! k" a! H2 [% G) _. i0 U
* P# ^" ^) k# u3 k0 W; \这种特性使得宏定义变得更加安全(因为每个操作数都只被计算一次,例如++运算)。例如计算最大值通常在C语言中被定义为这样的宏: + a! a. D1 h% x3 ^% Y
#define max(a,b) ((a) > (b) ? (a) : (b))
" M8 d" N* t' F! y" u
8 [" D. Y! b0 @% I6 j8 z1 }7 `但是其中的a和b可能会被计算两次,如果操作数带有副作用,则会产生错误的结果。在GNU C中,如果你知道了操作数的类型(假设为int),你可以这样安全的定义宏:
9 W2 A# y5 q8 y1 i% c: T$ I3 q#define maxint(a,b) \9 [( J* `+ |* E, _& G8 E( T- T
({int _a = (a), _b = (b); _a > _b ? _a : _b; })2 w) ]7 g9 |9 m1 L1 |
语句内嵌在常量表达式(例如枚举类型),位域尺寸或静态变量初始化中是不允许的。如果你不知道操作数的类型,你也可以使用typeof来获得类型。0 Y$ P# O% b0 Q
语句表达式内嵌在G++中并不支持,而且将来是否支持目前也不清楚(他们在某时被完全支持或者被抛弃掉,或者作为bug会一直存在)。就目前而言,语句内嵌表达式在默认情况下工作的并不好。- \8 y' t {# W+ c, E
此外,在C++中语句内嵌表达式还存在很多语义问题。如果你希望在C++中用语句内嵌表达式来代替内联函数(inline function),对象的析构处理可能会让你惊讶。例如:! q: f' J6 f- p3 N! R6 f! n0 t
#define foo(a) ({int b = (a); b + 3; })
+ `; S! R" d8 s并不等同于
6 l( d( F+ c+ {inline int foo(int a) { int b = a; return b + 3; }# B8 l# H( U! c
具体而言,当传递给foo的表达式的会引入临时对象的生成的时候,这些临时对象的析构在用宏时会早于用函数的情况。- F3 @4 }! F7 Y9 H
以上情况说明在用于C++代码的.h头文件中使用语句内联表达式并不是一个好主意。一些GNU C的库的某些版本中的使用语句内联表达式的头文件已经造成了这样的bug。 |
|