C++17起,多数内置运算符(如+、-、*、/、==、&&、=等)明确规定左操作数先于右操作数求值,函数实参也按从左到右顺序求值,但同一对象的无序读写或多次修改仍导致未定义行为。
在C++中,表达式求值方式和执行顺序不是简单地“从左到右”或“从右到左”能概括的,它由求值顺序(evaluation order)、运算符优先级(precedence)、结合性(associativity)以及序列点(sequence points)共同决定。尤其自C++17起,标准对多数内置运算符的求值顺序做了明确约束,大幅减少了未定义行为(UB)的发生可能。
很多人误以为“乘除优先于加减”意味着乘法一定先算——其实它只影响表达式如何被解析成树形结构。例如:
a + b * c 被解析为 a + (b * c),但 a 的求值时机并未被规定;C++17前,a 和 b * c 的子表达式求值顺序是未指定的(unspecified),甚至可能交错;C++17起,对于大多数内置二元运算符(如 +、-、*、/),左操作数在右操作数之前求值。
C++17将原本“未指定顺序”的情况,明确为左操作数先于右操作数求值,适用于以下常见运算符:
留短路语义,且左操作数一定先求值)⚠️例外:逗号运算符 , 和三目运算符 ?: 本就有序(逗号左→右;?: 条件先,然后仅一个分支),C++17未改变它们。
以前写 func(f(), g(), h()),参数调用顺序完全未指定。C++17起,所有函数实参按从左到右顺序求值(即 f() → g() → h()),且每个实参的求值与其副作用,在下一个实参开始求值前全部完成。
即使C++17加强了约束,以下情形仍属未定义行为(UB),务必避免:
基本原则:如果两个副作用(如修改变量)或一个副作用与一个读取作用在同一对象上,且它们之间没有明确的求值顺序关系,就构成UB。
基本上就这些。理解C++表达式求值,关键不是背规则,而是养成“副作用隔离”习惯:把有副作用的操作(如自增、IO、函数调用)拆到独立语句,或用临时变量显式控制顺序。C++17让很多常见代码更可预测,但没消除对逻辑清晰性的要求。