为什么在行尾需要分号?

问题描述 投票:9回答:7

为什么这样做:

a = []
a.push(['test']);
(function() {alert('poop')})()

但这给出了错误“数字不是函数”:

a = []
a.push(['test'])
(function() {alert('poop')})()

唯一的区别是第2行末尾的分号。我已经写了很长时间的JavaScript了。我知道自动分号插入,但我无法弄清楚会导致此错误的原因。

javascript syntax
7个回答
12
投票

看看这个链式函数调用的例子。

a.push(['test'])(function() {alert('poop')})()

看起来熟悉?这就是编译器/解释器查看代码的方式。

详情

这是用于描述调用表达式的语法的一部分。

CallExpression : 
	MemberExpression Arguments 
	CallExpression Arguments 
	CallExpression [ Expression ] 
	CallExpression . IdentifierName 

基本上每个组(...)被视为原始MemberExpression a.push的参数。

a.push (['test'])                // MemberExpression Arguments 
(function() {alert('poop')})     // Arguments  
()                               // Arguments 

或者更正式

CallExpression(
    CallExpression(  
        CallExpression(
            MemberExpression( a.push ),
            Arguments( (['test']) )
        ),
        Arguments( (function() {alert('poop')}) )
    ),
    Arguments( () )
)

3
投票

我不是Javascript专家(甚至是新手:),但是如果你将第二行和第三行结合起来,它仍然看起来语法上有效:

a.push(['test'])(function() {alert('poop')})()

那是试图将a.push(['test'])的结果作为函数处理,将函数传递给它...然后将结果作为函数调用。

我怀疑如果两个语句可以在语法上组合成一个语句,则需要使用分号,但这不是你想要的。


1
投票

因为

a.push(['test'])(function() {alert('poop')})()

是一个有效的JavaScript表达式。如果某些相邻的行可以粘在一起形成一个有效的JavaScript表达式,那么您的JavaScript引擎会将它们粘在一起。

虽然它是一个有效的JavaScript表达式,但a.push返回一个不是函数的数字,当你尝试调用一个不是函数的东西时,它会返回你看到的错误。


1
投票

a.push(['test'])将返回数组的长度,一个数字。如果没有分号,编译器将解释自调用函数的开括号,就像您尝试将该数字作为带参数的函数执行一样。假设它返回了7的长度,那么本质上发生的事情就像你写的那样:

7 (function() {alert('poop')})();

因此错误“数字不是函数”,因为它不知道如何调用7作为函数。


1
投票

有些情况下,white space不需要作为令牌分隔符,但仅用于提高可读性:

空格字符用于提高源文本可读性并将标记(不可分割的词汇单元)彼此分开,但在其他方面无关紧要。任何两个令牌之间可能会出现空白区域,但不能出现在任何其他类型的令牌中。

在这种情况下,a.push(['test'])(function() {alert('poop')})()之间的空白区域不适用于令牌分隔符,因此无关紧要。所以它等同于:

a.push(['test'])(function() {alert('poop')})()

由于a引用长度为0的空数组,因此调用a.push(['test'])会将一个元素追加到a并返回a.length的更新值,即1

1(function() {alert('poop')})()

剩下的就是历史。


0
投票

a.push(['test'])返回一个数字。

然后,您尝试将该数字作为函数调用function() {alert('poop')}作为唯一参数。

因此你的错误,number is not a function


0
投票

a.push()的结果是返回数组的大小,在本例中为1.围绕函数的括号使得Javascript解析器认为您正在尝试调用:

1( function() {alert('poop')} )()

或者您尝试使用匿名函数作为参数调用名为1的函数,然后将return作为函数执行。

© www.soinside.com 2019 - 2024. All rights reserved.