为什么在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 。