c++ - 为什么带逗号的三元运算符在真实情况下只计算一个表达式?

标签 c++ language-lawyer operator-precedence conditional-operator

我目前正在通过《C++ Primer》这本书学习 C++,书中的练习之一是:

Explain what the following expression does: someValue ? ++x, ++y : --x, --y

我们知道什么?我们知道三元运算符的优先级高于逗号运算符。对于二元运算符,这很容易理解,但是对于三元运算符,我有点挣扎。使用二元运算符“具有更高优先级”意味着我们可以在具有更高优先级的表达式周围使用括号,并且不会改变执行。

对于三元运算符,我会这样做:

(someValue ? ++x, ++y : --x, --y)

有效地产生相同的代码,这无助于我理解编译器将如何对代码进行分组。

但是,通过使用 C++ 编译器进行测试,我知道表达式可以编译,但我不知道 : 运算符本身可以代表什么。所以编译器似乎正确地解释了三元运算符。

然后我以两种方式执行程序:

#include <iostream>

int main()
{
    bool someValue = true;
    int x = 10, y = 10;

    someValue ? ++x, ++y : --x, --y;

    std::cout << x << " " << y << std::endl;
    return 0;
}

结果:

11 10

而另一方面 someValue = false 它会打印:

9 9

为什么 C++ 编译器会生成代码,对于三元运算符的真分支只增加 x,而对于三元运算符的假分支,它会减少两个 xy?

我什至像这样在真分支周围加上括号:

someValue ? (++x, ++y) : --x, --y;

但它仍然会导致 11 10.

最佳答案

作为 @Rakete在他们出色的回答中说,这很棘手。我想补充一点。

三元运算符必须具有以下形式:

logical-or-expression ? expression : assignment-expression

所以我们有以下映射:

  • someValue : 逻辑或表达式
  • ++x,++y : 表达式
  • ???是 assignment-expression --x, --y 还是只有 --x

其实只是--x,因为一个赋值表达式不能解析成两个用逗号隔开的表达式(根据C++的语法规则),所以--x, --y 不能被视为赋值表达式

这导致三元(条件)表达式部分如下所示:

someValue?++x,++y:--x

为了便于阅读,考虑将 ++x,++y 计算为 as-if 括号中的 (++x,++y ); ?: 之间包含的任何内容都将在条件 之后进行排序。 (我将在帖子的其余部分用括号括起来)。

并按此顺序评估:

  1. someValue?
  2. (++x,++y)--x(取决于 bool1 的结果。)

这个表达式然后被视为逗号运算符的左子表达式,右子表达式为 --y,如下所示:

(someValue?(++x,++y):--x), --y;

这意味着左侧是一个丢弃值表达式,意味着它肯定被评估,但是我们评估右侧并返回它。

那么当 someValuetrue 时会发生什么?

  1. (someValue?(++x,++y):--x) 执行并将 xy 递增为 1111
  2. 左表达式被丢弃(尽管增量的副作用仍然存在)
  3. 我们评估逗号运算符的右侧:--y,然后将 y 递减回 10

要“修复”该行为,您可以将 --x, --y 用括号分组,以将其转换为 主要表达式 assignment-expression*:

的有效条目
someValue?++x,++y:(--x, --y);

*这是一个相当有趣的长链,将 assignment-expression 连接回主表达式:

赋值表达式 ---(可以包括)--> 条件表达式 --> 逻辑或表达式 - -> 逻辑和表达式 --> 包含或表达式 --> 排除或表达式 --> 和表达式 --> 等式表达式 --> 关系表达式 --> 移位表达式 --> < em>additive-expression --> multiplicative-expression --> pm-expression --> cast-expression --> 一元表达式 --> 后缀表达式 --> 主表达式

关于c++ - 为什么带逗号的三元运算符在真实情况下只计算一个表达式?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47538906/

相关文章:

c++ - 为什么 std::priority_queue::emplace 是对数的?

c++ - 是否允许比较 static_assert 中静态类字段上的指针?

c# - 新的 C# 8.0 switch 表达式的运算符优先级是什么?

c++ - Visual Studio (2015) C 项目依赖项 - 必须定义 LNK1561 入口点

c++ - C++ 中的类 C 程序?

c++ - 如何在 cmake 生成的 visual studio 项目中包含目录?

python - 依赖 Python 函数参数评估顺序是否安全?

c++ - 使用 libcurl 更新表单中的字段值

c++ - decltype-specifier 表示的类型是什么类型,其表达式是类类型的临时对象的成员?

java - 给予括号优先