Чому виклик функції у Node.js REPL з) (працює?


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
Здається, .jsпомилка
REPL

8
До речі, кредит , де це пов'язано, це придумали в IRC (FreeNode #nodejs), по @miniml
Hyde

3
Perl має щось подібне з тієї ж причини: perl -ne '$x += $_; }{ print $x'. Дивіться приховані функції Перла
Адріана Пронка

Відповіді:


84

Здається, це помилка 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
Вони насправді пішли вперед і виправили це? Awwww, шкода, я дуже хотів би, щоб це почало культуру та стало функцією на всіх мовах. Скільки разів я набрав) (замість () поспішаю ... :))
geomagas

18
@geomagas Ви вважаєте, що function a)arg1, arg2( } ]arg2 + arg1[ return; {повинен бути правильним синтаксисом?
azz

40
Ні, не дуже. Власне, це був жарт.
geomagas

7
Колись була реалізація Lisp з опцією DWIM, яка автоматично виправляла неправильні написання та інші незначні помилки. en.wikipedia.org/wiki/DWIM
Barmar

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/isissue/5698

І проблема полягала в тому, що REPL додає висловлювання до паролів. Так

foo)(

стає

(foo)()

Фактичне пояснення можна знайти тут https://github.com/joyent/node/isissue/5698#issuecomment-19487718 .


75
І я вважав, що автоматичне вставлення крапки з комою погано.
Ерік Ліпперт
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.