运算符的优先级、结合性、操作数的求值顺序
看了⼏个微博,组织到⼀起记录下
⼀.运算符的优先级
在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;

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。