简单地说,在Java中对于复合赋值操作符会先对左侧操作数进行求值,随后对右侧操作数进行求值,最后将两个值进行运算并赋值给左侧操作数。(
JLS8 15.7.1 左侧操作数首先求值)
也就是说在 a += a *= a; 中,会先从左到右求出三个 a 的值(均为12),然后将后两个值相乘得到144,并赋值给 a,最后将第一个值(注意这里用的不是144而是最开始得到的12)和144相加得到156,并赋值给 a。最终 a = 156。
对于C++来说,在C++17之前,这是未定义行为。而在C++17中对于这种表达式的计算顺序是有规定的。在C++17中,复合赋值操作符右侧的操作数的计算及副作用保证在左侧的操作数之前完成,也就是说先进行 a *= a,将 a 赋值为144,然后再计算最左侧的 a(144)并与刚才得到的144相加得到288并赋值给 a ,最终 a = 288。
这跟编译器的解释规则相关,一般来说赋值从右往左。我个人觉得哈,这个问题真的没什么意义。没有人会在实际中写这种完全没有可读性的垃圾代码。