为什么在使用()调用Node.js REPL中的函数起作用?


191

为什么可以用经node.js测试的JavaScript调用此函数:

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

为什么最后一次呼叫“” hi)(有效?是node.js中的错误,V8引擎中的错误,正式未定义的行为还是对所有解释程序实际上有效的JavaScript?


1
在Ubuntu 13.04上的nodejs v0.6.19中可重现
mvp

1
在jsfiddle.net上进行的快速测试将显示它是无效的JavaScript。
克里斯托夫·

6
似乎是Node REPL错误,将这两行放在a内.js将导致语法错误
leesei 2013年

8
顺便说一句,应归功于这,这是@irml在irc(FreeNode #nodejs)上提出的
hyde

3
Perl也有类似的原因,其原因大致相同:perl -ne '$x += $_; }{ print $x'。请参阅Perl的隐藏功能
Adrian Pronk

Answers:


84

似乎是Node 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)
> 

27
他们实际上继续修复了吗?噢,可惜,我真的很想看到它开始一种文化,并成为所有语言的特色。我多少次输入)(而不是()急... :))
geomagas

18
@geomagas您认为function a)arg1, arg2( } ]arg2 + arg1[ return; {应该是有效的语法?
azz

40
不,不是。实际上,这是个玩笑。
geomagas 2013年

7
曾几何时,Lisp实现带有DWIM选项,可以自动纠正拼写错误和其他较小的错误。en.wikipedia.org/wiki/DWIM-
巴马(Barmar)

2
@geomagas,还有一些已经说干就干,想到了它- npm已经install isntall。也许您没有注意到:)
Eliran Malka 2014年

201

这是由于REPL如何评估输入,最终是:

(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,它将只包装{ ... }而不是所有输入:

  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';
  }

19
这是否意味着hi)(arg有效?写一些真正的WTF缠身的代码可能会被滥用;-)
琼斯医生

我仍然不明白为什么会这样。由于没有开放的括号,它不会产生语法错误吗?
彼得·奥尔森

2
hi)(arg成为(hi)(arg)-没有什么
比不上

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.