阅读本文的前提是,你能区分什么是表达式,什么是语句。还有明确运算符和运算数都是些啥东西。所谓的表达式就是一个JavaScript的“短语“,JavaScript的解释器可以计算它,从而生成一个值。表达式可以分为以下三种:

  • 1)直接量,如1.7是数字直接量、"JavaScript权威指南"是一个字符串直接量等。
  • 2)变量,直接量表达式的值就是这个直接量本身,变量表达式的值则是该变量所存放或引用的值。
  • 3)可以“合并”上面提到的“简单”表达式来创建为比较复杂的表达式。例如1.7是表达式,i也是表达式,下面的代码所示的同样也是(也可以叫)表达式:i + 1.7

运算数的个数:
可以根据运算符需要的运算数的个数对运算符进行分类,大多数运算符都是二元运算符,它们把两个“表达式”合并成一个复杂的表达式。简而言之,就是它有两个运算数。此外,JavaScript还支持大量的一元运算符,它能将一个表达式转换成另一个更复杂的表达式。如表达式-3中,运算符“-”就是一元运算符, 它执行操作是对运算数取反。JavaScript还支持三元操作符“?:”,它可以将三个表达式合并为一个复杂的表达式。

逗号运算符

逗号表达式的一般形式是:表达式1,表达式2,表达式3……表达式n。逗号表达式的求解过程是:先计算表达式1的值,再计算表达式2的值,……一直计算到表达式n的值。最后整个逗号表达式的值是表达式n的值。例子:

var a = 10, b = 20;
function CommaTest(){
  return a++, b++, 10;
}
var c = CommaTest();
alert(a); // 返回11
alert(b); // 返回21
alert(c); // 返回10

变量c的值是函数CommaTest返回的值,而a和b多加1了。
逗号运算符和函数调用运算符的冲突
在JavaScript中,函数调用确实是函数调用运算符。它很特殊,因为其它编程语言资料中多从来没有这个叫法的。函数调用运算符的第一个参数是一个函数名或者是一个引用函数的表达式,其后是括号()。括号中间可以是数目不定的运算数,这些运算数可以是任意的表达式,它们之间用逗号隔开。函数调用运算符将计算它的每一个运算数,第一个运算数指定为函数名(括号前),而括号中间的所有运算数的值将传递给这个函数作为函数的参数。

alert(2*5, 2*4); // 输出10

上面这段代码输出10,但是如果根据逗号运算符的原理来解释的话,那应该是输出8才对。为什么呢?

因为逗号运算符在JavaScript在的优先级是最底的,记住这一点非常有用。所以函数调用运算符将先于逗号运算符运行。结果alert函数输出第一个参数的值。将上面的代码修改成如下所示即可。

alert((2*5, 2*4)); // 返回8

逗号运算符和赋值运算符的冲突
在JavaScript中,逗号运算符的优先级比赋值运算符还要底。请看下面的代码。

var a = 20;
var b = ++a,10;
alert(b);

这段代码似乎不能运行,可能是由于赋值运算符优先于逗号表达式运行,如果将代码改成

var a = 20;
var b = (++a,10);
alert(b);

即可了。上面说到的“可能”我们这里做一下解释,这是本人的一些看法,不一定权威。
逗号运算符要求它的运算数是一个复杂的表达式或简单的表达式(如变量或直接量),但由于赋值运算符优先于逗号运算符执行,因此变成左边不是一个运算数或一个表达式,而是一个含有var关键字的语句。之前不能执行的代码可以看成如下代码:

var a = 20;
(var b = ++a),10;
alert(b);

语句中有表达式语句,但不是所有的语句都是表达式。

逗号运算符的用途

逗号表达式用的地方不太多,一般情况是在给循环变量赋初值时才用得到。所以程序中并不是所有的逗号都要看成逗号运算符,尤其是在函数调用时,各个参数是用逗号隔开的,这时逗号就不是逗号运算符。运算符使它两边的表达式以从左到右的顺序被执行,并获得右边表达式的值。, 运算符最普通的用途是在 for 循环的递增表达式中使用。例如:

for (i = 0; i < 10; i++, j++)
{
  k = i + j;
}

每次通过循环的末端时, for 语句只允许单个表达式被执行。逗号运算符被用来允许多个表达式被当作单个表达式,从而规避该限制。