看了几个微博,组织到一起记录下
一.运算符的优先级
在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;