C++的魔术,这个表达式 x = y - x + (y = x)

Magic in C++ with this expression x = y - x + (y = x)

本文关键字:表达式 魔术 C++      更新时间:2023-10-16

在 Python 中,您可以使用如下表达式轻松交换 2 个变量的值:

x, y = y, x

另一方面,在C++中,如果要交换 2 个变量的值,通常使用临时变量来存储其中一个变量的值,如下所示:

int var1 = 100;
int var2 = 200;
int temp = var1;
var1 = var2;
var2 = temp;

这很容易,但你必须编写大量代码。

我一直在关注的一位教授关于C++的讲座,他发现了一种新的紧凑方式,以一种神奇的方式交换 2 个变量的值:

int x = 200;
int y = 100;
x = y - x + (y = x);

这似乎令人难以置信,但它既适用于他正在使用的编译器,也适用于我的Apple LLVM version 6.0 (clang-600.0.56)

似乎表达式的解释方式如下:

  1. (y = x)//返回 x 的值
  2. -x + x = 0
  3. x = y + 0
  4. x = y
  5. y = x/
  6. /最后,y 得到 x 的初始值

如果我尝试在循环中交换某些变量的值,它似乎也可以工作:

for (int i = -10; i <= 10; i++) {
    for (int j = 10; j >= -10; j--) {
        int x = i, y = j;
        x = y - x + (y = x);
        std::cout << "x = " << x << "ny = " << y << 'n';
    }
}

我们已经看到交换已经完成,但我的编译器给了我这个警告:

主.cpp:28:22: 警告:未排序的修改和访问"Y" [-温序列]

如果有警告,我想这不是交换 2 个变量值的新标准方法,而只是这位教授的一个巧妙的解决方法。

为什么这种方法在C++中没有标准化,为什么它到底有效?

这是未定义的行为,不能保证有效。未指定表达式中参数的计算顺序。因此,允许在执行y = x分配之前或之后访问减法的y值。

这样指定它的原因是允许优化的灵活性。

它"有效"是因为你运气不好。

是的,我说倒霉。代码具有未定义的行为;它似乎"工作"的事实意味着您的编译器可以让您摆脱下次运行它时可能失败的代码。

x = y - x + (y = x);

对象y被访问两次,一次读取其值,一次为其赋值。该语言没有定义这些访问发生的顺序 - 或者根本不以任何顺序发生。正如警告消息所说,它们是"未排序的"。可能的行为不限于两个可能的订单;行为是完全未定义的。

此外,加法和/或减法可能会溢出,这是未定义行为的另一个潜在来源。

这, 'x = y

- x + (y = x);, 是 C 语言中未定义的行为 - C99 规范列出的一种未定义行为是:

在两个序列点之间,一个对象被多次修改,或者 被修改并读取先前的值,而不是确定 要存储的值

此表达式中唯一的序列点是表达式的开头和完整表达式的结尾。