Почему вызов функции в REPL с Node.js) (работает?

191

Почему можно вызвать функцию в 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, официально неопределенное поведение или действительно допустимый JavaScript для всех интерпретаторов?

Хайд
источник
1
Воспроизводимый в nodejs v0.6.19 на Ubuntu 13.04
mvp
1
быстрый тест на jsfiddle.net покажет вам, что это неверный JavaScript.
Кристоф
6
Кажется, это ошибка Node REPL, когда две строки в .jsбудут вызывать синтаксическую ошибку
leesei
8
Кстати, поверьте, это произошло в irc (FreeNode #nodejs), @miniml
hyde
3
Perl имеет нечто подобное по той же причине: perl -ne '$x += $_; }{ print $x'. См. Скрытые возможности Perl
Адриан Пронк

Ответы:

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)
> 
leesei
источник
27
Они на самом деле пошли дальше и исправили это? Жаль, я бы очень хотел, чтобы это зародило культуру и стало фичой на всех языках. Сколько раз я набирал) (вместо () в спешке ... :))
geomagas
18
@geomagas Вы думаете, function a)arg1, arg2( } ]arg2 + arg1[ return; {должен быть правильный синтаксис?
Azz
40
Нет, не совсем. На самом деле, это была шутка.
Geomagas
7
Когда-то была реализация Lisp с опцией DWIM, которая автоматически исправляла ошибки и другие мелкие ошибки. ru.wikipedia.org/wiki/DWIM
Бармар,
2
@geomagas, ну, некоторые уже пошли дальше и подумали об этом - npmесть install и isntall . держу пари, что вы не заметили :)
Элиран Малка
201

Это связано с тем, как REPL оценивает входные данные, которые в конечном итоге таковы:

(hi)()

Добавлены дополнительные скобки, чтобы заставить его быть выражением :

  // 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)- ничто не имеет себе равных
SheetJS
60

Была ошибка, поднятая 4 месяца назад, для этой проблемы https://github.com/joyent/node/issues/5698

И проблема была в том, что REPL заключает в себе заявления с паренами. Так

foo)(

становится

(foo)()

Актуальное объяснение можно найти здесь https://github.com/joyent/node/issues/5698#issuecomment-19487718 .

thefourtheye
источник
75
И я думал, что автоматическая вставка точки с запятой была плохой.
Эрик Липперт