Почему вызов функции в Node.js REPL с) (работа?
Почему можно вызвать функцию в JavaScript, подобную этой, протестированной с помощью node.js:
~$ node
> function hi() { console.log("Hello, World!"); };
undefined
> hi
[Function: hi]
> hi()
Hello, World!
undefined
> hi)( // WTF?
Hello, World!
undefined
>
Почему работает последний вызов hi)(
? Это ошибка в node.js, ошибка в двигателе V8, официальное поведение undefined или действительно действующий JavaScript для всех интерпретаторов?
Ответы
Ответ 1
Кажется, это ошибка 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)
>
Ответ 2
Это связано с тем, как REPL оценивает ввод, который в конечном итоге выглядит как:
(hi)()
Дополнительные скобки добавляются чтобы заставить это быть выражением:
// First we attempt to eval as expression with parens.
// This catches '{a : 1}' properly.
self.eval('(' + evalCmd + ')',
// ...
Цель состоит в том, чтобы рассматривать {...}
как Object
литералы /initialisers, а не как блок.
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 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 an
// unexpected end of input.
evalCmd = evalCmd + '\n';
}
Ответ 3
Была ошибка, поднятая 4 месяца назад, для этой проблемы https://github.com/joyent/node/issues/5698
И проблема была в том, что REPL заключает в себя инструкции с помощью parens. Так
foo)(
становится
(foo)()
Фактическое объяснение можно найти здесь https://github.com/joyent/node/issues/5698#issuecomment-19487718.