🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
从最简单的运算符加号(+)说起,加号(+)是个二元运算符——也就是说,**加号只把两个数联接起来,从来不把第三个或者更多的联接起来**。 因此,“1加2加3” 在计算机中被表述为: ~~~ (1 + 2) + 3 // a ~~~ 或者 ~~~ 1 + (2 + 3) // b ~~~ 虽然我们通常写做 `1 + 2 + 3`,但是并不意味这它和我们数学中的 1+2+3 是等价的。 那么数学中的 1+2+3 到底表示的是 a 呢,还是 b 呢? 如果计算机的求值是左结合的,那么此表达式等价于第一种a; 如果是右结合的,那么此表达式等价于第二种b。 `1 + 2 + 3` 简单的理解就是 “把1、2、3加在一起”, 确实,在我们接触到的数学里面,就是把三个数加起来。 但是在编程语言中,却不仅仅这样。 就像前面说的那样,+号无法操作三个或者更多的数,参与加法运算的只能是两个数。 顺便说一句,正号、负号是一元运算符,虽然它们和二元运算符加、减用相同的符号, 但是他们却是不同的,所以不要想当然的认为 +4 就等价于 0+4,其实它们不是等价的, **+4 是一个整数,但是 0+4 是一个加法表达式,这个表达式的求值结果正好是 +4**。 在 java 中,我们可以写 `short a = +4`,但是当我们写 `short a = 0 + 4` 时则产生一个警告。 还有一个其它例子,同样是关于 short 的, ~~~ short b = 1; short b = b + 4; // 警告 short b += 4; // 无警告 ~~~ 那么 `1 + 2 + 3` 是如何运算的呢? 在冯诺依曼体系架构的编程语言中, 这里有一个副作用——我习惯称那些“计算机的运算过程与程序员的大脑思考过程不一样时,则称为副作用”(虽然书本里面没有这么写过,但我一向这么认为), **本来你以为会是这样,结果计算机偏偏就不是这样做的,我称他为副作用。** 如果看过前面的『[语句与表达式](http://justjavac.com/codepuzzle/2012/10/28/codepuzzle-expression-and-statement.html)』,这可以这么理解: 1 + 2 是一个表达式,它的返回值是 3。 这个表达式的返回值再参加到另一个表达式中 3 + 3,最后得出结果6。 我们用语句(Statement)来改写这段代码: ~~~ // 计算 1 + 2 + 3 var a = 1 + 2; var b = b + 3; ~~~ 如果我们用 lisp 语言对这个表达式求值,则没有副作用。 ~~~ (+ (+ 1 2) 3) ~~~ 如果你还没有懂,或者这个例子太有特殊性,那么我们换一个 ~~~ 5 > 4 > 3 ~~~ 在数学中,这个算式的值为 true。当我们用C语言来写这段代码,它返回的确实 false。 原因和上面的一样,大于号(>)是二元运算,它无法直接比较三个数,5 > 4 返回的结果是 true, 当用 true 和 3 比较时,true 被转换称 1,也就是 `1 > 3`,最终的结果自然就是 false 了。 总之,回归到了『[语句与表达式](http://justjavac.com/codepuzzle/2012/10/28/codepuzzle-expression-and-statement.html)』篇的那个观点:在编程语言中 **每个表达式都有一个值**。 编程语言中的运算符和数学中的运算器虽然一样,但是它们却并不等同。 **当你写程序时,要写给人看; 当你调试程序时,要学会用计算机的方式思考代码的含义。** 我习惯于把运算符理解为函数,比如 2 + 5 其实就是 add(2, 5) 或者 2.add(5)。 难道我会偷偷的告诉你 “其实很多语言都是这么做的”。