Perché chiamare una funzione nel REPL di Node.js con) (funziona?


191

Perché è possibile chiamare la funzione in JavaScript in questo modo, testata con node.js:

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

Perché l'ultima chiamata hi)(funziona? È bug in node.js, bug nel motore V8, comportamento ufficialmente indefinito o JavaScript effettivamente valido per tutti gli interpreti?


1
riproducibile in nodejs v0.6.19 su Ubuntu 13.04
mvp

1
un rapido test su jsfiddle.net ti mostrerà che non è JavaScript valido.
Christophe,

6
Sembra essere un bug REPL del nodo, mettere le due righe in un .jscauserà un errore di sintassi
leesei,

8
A proposito, credito dove è dovuto, questo è venuto su irc (FreeNode #nodejs), di @miniml
hyde,

3
Perl ha qualcosa di simile per la stessa ragione: perl -ne '$x += $_; }{ print $x'. Guarda le funzioni nascoste di Perl
Adrian Pronk l'

Risposte:


84

Sembra essere un bug REPL del nodo, mettere queste due righe in un .jscauserà un errore di sintassi.

function hi() { console.log("Hello, World!"); }
hi)(

Errore:

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

Numero presentato # 6634 .

Riprodotto su v0.10.20.


v0.11.7 hanno risolto questo problema.

$ 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
In realtà sono andati avanti e l'hanno risolto? Oh, peccato, mi piacerebbe davvero vederlo iniziare una cultura e diventare una caratteristica in tutte le lingue. Quante volte ho digitato) (invece di () in fretta ... :))
geomagas,

18
@geomagas Pensi che function a)arg1, arg2( } ]arg2 + arg1[ return; {dovrebbe essere una sintassi valida?
azz

40
No, non proprio. In realtà, era uno scherzo.
geomagas,

7
C'era una volta un'implementazione di Lisp con un'opzione DWIM che correggeva automaticamente errori di ortografia e altri errori minori. en.wikipedia.org/wiki/DWIM
Barmar l'

2
@geomagas, beh, alcuni sono già andati avanti e ci hanno pensato - npmha install e isntall . scommetto che non l'hai notato :)
Eliran Malka

201

È dovuto al modo in cui REPL valuta l'input, che è in definitiva come:

(hi)()

Le parentesi aggiuntive vengono aggiunte per forzare ad essere un'espressione :

  // First we attempt to eval as expression with parens.
  // This catches '{a : 1}' properly.
  self.eval('(' + evalCmd + ')',
      // ...

L'intento è di trattare {...}come Objectletterali / inizializzatori piuttosto che come un blocco .

var stmt = '{ "foo": "bar" }';
var expr = '(' + stmt + ')';

console.log(eval(expr)); // Object {foo: "bar"}
console.log(eval(stmt)); // SyntaxError: Unexpected token :

E, come menzionato da Leesei, questo è stato modificato per 0.11.x, che andrà a{ ... } capo invece di tutto l'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';
  }

19
Significa che hi)(argfunzionerà? Ciò potrebbe essere abusato di scrivere del codice veramente dominato dalla WTF ;-)
Doctor Jones,

Continuo a non capire il motivo per cui dovrebbe funzionare. Non farebbe un errore di sintassi a causa della paren aperta senza eguali?
Peter Olson,

2
hi)(argdiventa (hi)(arg)- niente ineguagliato
SheetJS

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.