Mengapa memanggil fungsi dalam Node.js REPL with) (berfungsi?

191

Mengapa mungkin memanggil fungsi dalam JavaScript seperti ini, diuji dengan node.js:

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

Mengapa panggilan terakhir hi)(, bekerja? Apakah itu bug di node.js, bug di mesin V8, perilaku resmi yang tidak terdefinisi, atau JavaScript yang benar-benar valid untuk semua penerjemah?

Hyde
sumber
1
direproduksi dalam nodejs v0.6.19 di Ubuntu 13.04
mvp
1
tes cepat di jsfiddle.net akan menunjukkan kepada Anda bahwa itu adalah JavaScript yang tidak valid.
Christophe
6
Tampaknya menjadi bug Node REPL, menempatkan dua baris di .jsakan menyebabkan kesalahan sintaksis
leesei
8
Btw, kredit di mana sudah waktunya, ini muncul di irc (FreeNode #nodejs), oleh @miniml
hyde
3
Perl memiliki sesuatu yang mirip untuk banyak alasan yang sama: perl -ne '$x += $_; }{ print $x'. Lihat fitur Tersembunyi dari Perl
Adrian Pronk

Jawaban:

84

Tampaknya bug Node REPL, menempatkan dua baris ini dalam .jsakan menyebabkan kesalahan sintaksis.

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

Kesalahan:

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

Masalah yang disampaikan # 6634 .

Direproduksi pada v0.10.20.


v0.11.7 memperbaiki ini.

$ 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
sumber
27
Mereka benar-benar pergi ke depan dan memperbaikinya? Sayang sekali, saya benar-benar ingin melihatnya memulai budaya dan menjadi fitur dalam semua bahasa. Berapa kali saya mengetik) (bukannya () terburu-buru ... :))
geomagas
18
@geomagas Anda pikir function a)arg1, arg2( } ]arg2 + arg1[ return; {seharusnya sintaks yang valid?
azz
40
Tidak terlalu. Sebenarnya, itu hanya lelucon.
geomagas
7
Sekali waktu ada implementasi Lisp dengan opsi DWIM yang secara otomatis memperbaiki kesalahan ejaan dan kesalahan kecil lainnya. en.wikipedia.org/wiki/DWIM
Barmar
2
@geomagas, yah, beberapa sudah berjalan dan memikirkannya - npmsudah install dan isntall . yakin Anda belum menyadarinya :)
Eliran Malka
201

Itu karena bagaimana REPL mengevaluasi input, yang pada akhirnya adalah:

(hi)()

Tanda kurung tambahan ditambahkan untuk memaksanya menjadi Ekspresi :

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

Tujuannya adalah untuk memperlakukan {...}sebagai Objectliteral / inisialisasi daripada sebagai blok .

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

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

Dan, seperti yang disebutkan leesei, ini telah diubah untuk 0.11.x, yang hanya akan membungkus{ ... } daripada semua 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';
  }
Jonathan Lonowski
sumber
19
Apakah itu berarti itu hi)(argakan berhasil? Itu bisa disalahgunakan untuk menulis beberapa kode yang benar-benar dikendarai WTF ;-)
Doctor Jones
Saya masih tidak mengerti mengapa itu akan berjalan. Bukankah itu akan membuat kesalahan sintaksis karena paren terbuka yang tidak cocok?
Peter Olson
2
hi)(argmenjadi (hi)(arg)- tidak ada yang tak tertandingi
SheetJS