运算符的结合方向
转载说明:这篇文章是以前在网上看到的,它以很容易理解的方式解释了什么是运算符的优先级,当时觉得写的不错,因此看完之后保存了下来,今天拿出来与大家分享。由于已不知道它的作者和出处,因此无法标明了,请作者见谅。
在标准C语言的文档里,对操作符的结合性并没有作出非常清楚的解释。一个满分的回答是:它是仲裁者,在几个操作符具有相同的优先级时决定先执行哪一个。
每个操作符拥有某一级别的优先级,同时也拥有左结合性或右结合性。优先级决定一个不含括号的表达式中操作数之间的“紧密”程度。例如,在表达式a*b+c中,乘法运算的优先级高于加法运算符的优先级,所以先执行a*b,而不是b+c。
但是,许多操作符的优先级是相同的。这时,操作符的结合性就开始发挥作用了。在表达式中如果有几个优先级相同的操作符,结合性就起仲裁的作用,由它决定哪个操作符先执行。像下面这个表达式:
int a, b=1, c=2;
a=b=c;
我们发现,这个表达式只有赋值符,这样优先级就无法帮助我们决定哪个操作先执行,是先执行b=c呢,还是先执行a=b?如果按前者,a的结果为2,如果按后者,a的结果为1。
所有的赋值符(包括复合赋值)都具有右结合性,就是说在表达式中最右边的操作最先执行,然后从右到左依次执行。这样,c先赋值给b,然后b再赋值给a,最终a的值是2。类似地,具有左结合性的操作符(如位操作符“&”和“|”)从左至右依次执行。
所以,遇到一个表达式时,先根据运算符的优先级将表达式拆分成几个子表达式,然后在每个子表达式中根据运算符的结合性来进一步确定执行的顺序。
【特殊情况】 看下面这个C表达式:
a+++b;
结果是(a++)+b还是a+(++b)呢?答案是前者,因为在遇到这种情况时,C编译器会将表达式按照从左往右的顺序尽量匹配出一个子表达式,所以当C
编译器遇到a且后面有3个+号时,会首先尝试匹配a+++,当然这是没有意义的,所以接着尝试a++发现这是有意义的匹配,因此确定a++为第一个子表达
式,然后再接着匹配+号和b。因此结果是(a++)+b。
这种情况不能算是上面所讨论的运算符优先级和结合性的特例,因为它发生在优先级和结合性判断之前,但确实具有一些迷惑性,所以建议在遇到这种情况时给其加上括号,以增加可读性且避免可能发生的不必要的错误。