为什么在Node.js REPL中调用一个函数)(工作?

为什么可以像这样在JavaScript中调用函数,并使用node.js进行testing:

~$ node > function hi() { console.log("Hello, World!"); }; undefined > hi [Function: hi] > hi() Hello, World! undefined > hi)( // WTF? Hello, World! undefined > 

为什么最后一次调用hi)( ,工作吗?是node.js中的bug,V8引擎中的错误,官方未定义的行为,还是所有解释器实际上都有效的JavaScript?

似乎是一个节点的REPL错误,把这两行在.js将导致语法错误。

 function hi() { console.log("Hello, World!"); } hi)( 

错误:

 SyntaxError: Unexpected token ) at Module._compile (module.js:439:25) at Object.Module._extensions..js (module.js:474:10) at Module.load (module.js:356:32) at Function.Module._load (module.js:312:12) at Function.Module.runMain (module.js:497:10) at startup (node.js:119:16) at node.js:901:3 

提交的问题#6634 。

转载于v0.10.20。


v0.11.7有这个固定的。

 $ nvm run 0.11.7 Running node v0.11.7 > function hi() { console.log("Hello, World!"); } undefined > hi)( SyntaxError: Unexpected token ) at Object.exports.createScript (vm.js:44:10) at REPLServer.defaultEval (repl.js:117:23) at REPLServer.b [as eval] (domain.js:251:18) at Interface.<anonymous> (repl.js:277:12) at Interface.EventEmitter.emit (events.js:103:17) at Interface._onLine (readline.js:194:10) at Interface._line (readline.js:523:8) at Interface._ttyWrite (readline.js:798:14) at ReadStream.onkeypress (readline.js:98:10) at ReadStream.EventEmitter.emit (events.js:106:17) > 

这是由于REPL如何评估input,最终如下:

 (hi)() 

附加的括号被添加来强制它成为一个expression式

  // First we attempt to eval as expression with parens. // This catches '{a : 1}' properly. self.eval('(' + evalCmd + ')', // ... 

其目的是将{...}视为Object字面值/ 初始值,而不是块 。

 var stmt = '{ "foo": "bar" }'; var expr = '(' + stmt + ')'; console.log(eval(expr)); // Object {foo: "bar"} console.log(eval(stmt)); // SyntaxError: Unexpected token : 

而且,正如leesei所提到的,这已经被改为0.11.x, 它将包装{ ... }而不是所有的input:

  if (/^\s*\{/.test(evalCmd) && /\}\s*$/.test(evalCmd)) { // It's confusing for `{ a : 1 }` to be interpreted as a block // statement rather than an object literal. So, we first try // to wrap it in parentheses, so that it will be interpreted as // an expression. evalCmd = '(' + evalCmd + ')\n'; } else { // otherwise we just append a \n so that it will be either // terminated, or continued onto the next expression if it's an // unexpected end of input. evalCmd = evalCmd + '\n'; } 

有一个错误提出了4个月前,对于这个问题https://github.com/joyent/node/issues/5698

问题是因为,REPL包含了parens的陈述。 所以

 foo)( 

 (foo)() 

实际解释可以在这里findhttps://github.com/joyent/node/issues/5698#issuecomment-19487718 。