c – 赋值运算符中的序列点
作者:互联网
让我们以具体的复合赋值运算符^ =为例. This stackoverflow page表示在评估^ =之后可能尚未对左操作数进行修改,因此使得代码a ^ = b ^ = a ^ = b未定义的行为.但事实似乎并非如此.该标准在5.17 [expr.ass]中说明了这一点
In all cases, the assignment is sequenced after the value computation
of the right and left operands, and before the value computation of
the assignment expression.
本声明中有两个关键点. 1)主题分配是指什么?在我看来,它仅指左操作数的修改. 2)赋值表达式的值计算是指什么? cppreference说它指的是返回对修改过的对象的引用(强调我的).
作为结论,在评估^ =之后,左操作数应该已经被修改,这与大多数人的想法相矛盾.我在这里错过了什么吗?
解决方法:
您链接到C问题.但是,这是无关紧要的,因为C和C是不同的语言.
此外,从C11和C11开始,序列点不再存在;相反,存在之前,未经测序和不确定测序的关系.
在那句话中:
>赋值意味着写入a的内存位置.
>表达式的值计算意味着计算该表达式的值. (例子 – 2 2的值是4,值计算是确定4是值的过程).
这里有两个值计算:a ^ b和a =(结果).
在引用的文本中,对于a = a ^ b,事情必须按以下顺序发生:
>从a和b中检索值(按任意顺序),并确定存储结果的内存位置(分别为右和左操作数的值计算)
>将结果存储在(分配)中.涉及a ^ b的值计算,这在引用中没有提到,但显然必须在存储之前计算结果
>执行赋值表达式的值计算.这意味着放弃存储在为周围表达式准备的值(值计算).
你是对的,2和3似乎是“向后”,而不是你可能在纸上做的事情.但请记住,一般来说,y与x = y的值不同.赋值表达式的值与存储在x中的值相同. (例如:int x; double y =(x = 6.5); – 则y为6,而不是6.5).所以我们可以通过将结果存储在a中然后提供结果来实现.
标签:compound-assignment,c,assignment-operator,sequence-points 来源: https://codeday.me/bug/20191008/1873502.html