运算符的优先级、结合性、操作数的求值顺序

看了几个微博,组织到一起记录下

一.运算符的优先级

在C++ Primer一书中,对于运算符的优先级是这样描述的:
 Precedence specifies how the operands are grouped. It says nothing about the order in which the operands are evaluated.
意识是说优先级规定操作数的结合方式,但并未说明操作数的计算顺序。举个例子:
6+3*4+2
如果直接按照从左到右的计算次序得到的结果是:38,但是在C/C++中它的值为20。
因为乘法运算符的优先级高于加法的优先级,因此3是和4分组到一起的,并不是6与3进行分组。这就是运算符优先级的含义。

一共有十五个优先级:

1 () [] . ->

2 ! ~ -(负号) ++ – &(取变量地址)* (type)(强制类型) sizeof

3 * / %

4 + -

5 >> <<

6 > >= < <=

7 == !=

8 & (按位与)

9 ^

10 |

11 &&

12 ||

13 ?:

14 = += -= *= /= %= |= ^= &= >>= <<=

15 ,

就着多吧 结合性:2 13 14 是从右至左 其他都是 从左至右

括号成员第一; //括号运算符 成员运算符. ->

全体单目第二; //所有的单目运算符比如++、 –、 +(正)、 -(负) 、指针运算*、&乘除余三,加减四; //这个”余”是指取余运算即%

移位五,关系六; //移位运算符:<< >> ,关系:> < >= <= 等

等于(与)不等排第七; //即== 和!=

位与异或和位或; //这几个都是位运算: 位与(&)异或(^)位或(|)

“三分天下”八九十;

逻辑或跟与; //逻辑运算符:|| 和 &&

十二和十一; //注意顺序:优先级(||) 底于 优先级(&&)

条件高于赋值, //三目运算符优先级排到13 位只比赋值运算符和”,”高

逗号运算级最低! //逗号运算符优先级最低

注意:加减乘除高于移位操作

二.运算符的结合性

Associativity specifies how to group operators at the same precedence level.


结合性规定了具有相同优先级的运算符如何进行分组。

举个例子:

a=b=c=d;

由于该表达式中有多个赋值运算符,到底是如何进行分组的,此时就要看赋值运算符的结合性了。因为赋值运算符是右结合性,因此该表达式等同于(a=(b=(c=d))),而不是(a=(b=c)=d)这样进行分组的。

同理如m=a+b+c;

等同于m=(a+b)+c;而不是m=a+(b+c);

三.操作数的求值顺序

在C/C++中规定了所有运算符的优先级以及结合性,但是并不是所有的运算符都被规定了操作数的计算次序。在C/C++中只有4个运算符被规定了操作数的计算次序,它们是&&,||,逗号运算符(,),条件运算符(?:)。

  • 运算符&&和运算符||

    首先对左侧操作数求值,只在需要时才对右侧操作数求值。

  • 运算符?:

    有三个操作数据:在a?b:c中,操作数a首先被求值,根据a的值再求操作数b或c的值。

  • 逗号运算符

    首先对左侧操作数求值,然后该值被“丢弃”,再对左侧操作数求值。

from 《Essential C++》
逗号运算符的求值顺序是从左到右。
分割函数参数的逗号并不是逗号运算符,因此其求值顺序是未定义,不确定的。例如f(x, y); 不能保证其中x, y求值的先后。而g((x, y)); 的参数只有一个(x, y),这是一个逗号表达式,表达式的值就是y的值。同样的:

int x = f1() + f2();

中,对f1与f2调用的先后顺序也是不确定的。
下面这种从数组x中复制前n个元素到数组y中的做法是不正确的,因为它对求值顺序作了太多的假设:

while (i < n)
    y[i] =x [i++];

问题出在哪里呢?上面的代码假设y[i]的地址将在l的自增操作执行之前被求值,这一点并没有任何保证!在C语言的某些实现上,有可能在自增之前被求值;而在另外一些实现上,有可能与此相反。同样道理,下面这种版本的写法与前类似,也不正确:

while (i < n)
    y[i++]=x[i];

另一方面,下面这种写法却能正确工作:

while (i < n) {
    y[i] = x[i];
    i++;
}

当然,这种写法可以简写为:

for (i=0; i<n; i++)
    y[i] = x[i];

楚操作数的求值顺序和运算符的结合性 无关
如m=f1()+f2();
在这里是先调用f1(),还是先调用f2()?不清楚,不同的编译器有不同的调用顺序,甚至相同的编译器不同的版本会有不同的调用顺序。只要最终结果与调用次序无关,这个语句就是正确的。这里要分清楚操作数的求值顺序和运算符的结合性这两个概念,可能有时候会这样去理解,因为加法操作符的结合性是左结合性,因此先调用f1(),再调用f2(),这种理解是不正确的。结合性是确定操作符的对象,并不是操作数的求值顺序。

同理2+3*4+5;

它的结合性是(2+(3*4))+5,但是不代表3*4是最先计算的,它的计算次序是未知的,未定义的。

比如3*4->2+3*4->2+3*4+5

以及2->3*4->2+3*4->2+3*4+5和5->3*4->2+3*4->2+3*4+5这些次序都是有可能的。虽然它们的计算次序不同,但是对最终结果是没有影响的。


printf计算参数的时候是从右到左压栈的

*(ptr++) += 123; //  ++在后面表示计算后加加
//等效于 
*ptr +=123;
ptr++;
//等效于
*ptr = *ptr + 123;
ptr++;


*(++ptr) += 123; // ++在后面表示计算前加加
//等效于
ptr++;
*ptr +=123;
//等效于
ptr++;
*ptr = *ptr + 123;
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值