虽然文章标题是『语句与表达式』,在这篇文章中,我将陈述一个观点 **每个表达式都有一个值**。 在此之外,也会继续表述这个『[代码之谜](http://justjavac.com/codepuzzle/2012/09/25/codepuzzle-introduction.html)』系列的主题——数学与计算机之间被经常忽略的矛盾。
简单的讲
* "表达式"(expression)是一个单纯的运算过程,总是有返回值;
* "语句"(statement)是执行某种操作,没有返回值。
使用表达式也是函数式编程语言所提倡的,而传统命令式编程语言都是语句的堆砌。
表达式和语句如何区分呢? 最简单最直观的鉴别方法就是, **后面有分号的是语句**, 这是一个充分条件而不是必要条件。 有分号,就是语句;没有分号,就不一定了,也可能是语句,也可能是表达式。
在动态语言——比如javascript——中是通过上下文来区分这两者的。
假如如果 `function foo(){}` 在一个赋值表达式的一部分,则认为它是一个表达式。 **表达式的一部分,也是表达式**。 而如果 `function foo(){}` 被包含在一个函数体内,或者位于程序中,则将它作为一个语句。
~~~
function foo(){}; // 声明,因为它是程序的一部分
var bar = function foo(){}; // 表达式,因为它是赋值表达式的一部分
new function bar(){}; // 表达式,因为它是New表达式的一部分
(function(){
function bar(){}; // 声明,因为它是函数体的一部分
})();
~~~
还有一种不那么显而易见的表达式,就是被包含在一对圆括号中—— `(function foo(){})`。 将这种形式看成表达式同样是因为上下文的关系: (和)构成一个分组操作符,而 **分组操作符只能包含表达式**:
~~~
(function foo(){}); // 函数表达式:注意它被包含在分组操作符中
(var x = 5); // error! 分组操作符只能包含表达式,不能包含语句(这里的var就是语句)
~~~
今天突然有人问我:
~~~
alert(eval(data));
~~~
为什么会报错呢?data 是一个对象,按理说应该会弹出 Object[Object] 啊。 这是因为,当我们写
~~~
{"username" : "justjavac"}
~~~
时,它并不是一个对象。 因为我们知道有一种表示数据的方法叫做 json(javascript对象表示法), 所以想当然的认为这应该是一个对象。 其实,在大部分编程语言中,大括号({})表示的不是对象,而是代码块,这段代码其实等价于
~~~
{
"username" : "justjavac"
}
~~~
很显然,`"username" : "justjavac"` 并不是合法的语句。 然而解决方法也很简单,就是添加括号——分组操作符
~~~
({"username" : "justjavac"})
~~~
这样就构成了一个合法的表达式,当我们进行 json 对象解析的时候可以写如下代码:
~~~
eval('(' + json + ')')
~~~
**在表达式中,只能存在表达式,不能存在语句。**
例如表达式
~~~
(var a = 4) + 4;
~~~
这段代码将产生一个错误,因为 `var a = 4` 是一个语句, 而不是表达式—— **对表达式求值必须返回值,但对语句求值则未必有返回值**。
类似的
~~~
if (var a = 0) {}
~~~
也产生错误,因为 `var a = 0` 是一条语句,而 **语句没有返回值**。if 语句的语法结构为
~~~
if (expression) {
statement;
statement;
……
}
~~~
因此
~~~
if (var a = 0) {}
~~~
是错误的,但是
~~~
if (true) {
var a = 0;
}
~~~
则是正确的。
最后重申一下,**每个表达式都有一个值**。 理解了这个,就可以很容易的理解 FP(函数式编程)的一些核心思想了。