เหตุใดการเรียกใช้ฟังก์ชันใน 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 ที่ไม่ถูกต้อง
Christophe

6
ดูเหมือนว่าจะเป็นข้อผิดพลาดของ Node REPL การใส่สองบรรทัดใน a .jsจะทำให้เกิดข้อผิดพลาดทางไวยากรณ์
leesei

8
Btw, เครดิตที่ถึงกำหนด, สิ่งนี้เกิดขึ้นที่ irc (FreeNode #nodejs) โดย @miniml
hyde

3
Perl มีสิ่งที่คล้ายกันมากด้วยเหตุผลเดียวกัน: perl -ne '$x += $_; }{ print $x'. ดูคุณสมบัติที่ซ่อนของ Perl
Adrian Pronk

คำตอบ:


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
พวกเขาไปข้างหน้าและแก้ไขมันจริงหรือ เอาล่ะฉันอยากจะเห็นมันเริ่มวัฒนธรรมและกลายเป็นคุณสมบัติในทุกภาษา ฉันพิมพ์กี่ครั้ง) (แทนที่จะรีบ) ... ))
geomagas

18
@geomagas คุณคิดว่าfunction a)arg1, arg2( } ]arg2 + arg1[ return; {ควรเป็นไวยากรณ์ที่ถูกต้องหรือไม่
azz

40
ไม่ไม่ได้จริงๆ จริงๆแล้วมันเป็นเรื่องตลก
geomagas

7
กาลครั้งหนึ่งมีการใช้เสียงกระเพื่อมกับตัวเลือก DWIM ที่แก้ไขการสะกดผิดและข้อผิดพลาดเล็กน้อยอื่น ๆ โดยอัตโนมัติ en.wikipedia.org/wiki/DWIM
Barmar

2
@geomagas ดีบางคนแล้วเดินไปข้างหน้าและคิดเกี่ยวกับมัน - npmได้และinstall isntallเดิมพันที่คุณไม่ได้สังเกต :)
Eliran Malka

201

มันเป็นเพราะวิธีการที่ REPL ประเมินค่าอินพุตซึ่งท้ายที่สุดก็คือ:

(hi)()

วงเล็บเพิ่มเติมถูกเพิ่มเพื่อบังคับให้เป็นExpression :

  // 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'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 ;-) อย่างแท้จริง
Doctor Jones

ฉันยังไม่เข้าใจว่าเพราะเหตุใดจึงทำงาน มันจะไม่ทำให้เกิดข้อผิดพลาดทางไวยากรณ์หรือไม่เนื่องจากมีการเปิด paren ที่ไม่ตรงกันหรือไม่
Peter Olson

2
hi)(argกลายเป็น(hi)(arg)- ไม่มีอะไรที่ไม่ตรงกัน
SheetJS

60

มีข้อผิดพลาดเกิดขึ้น 4 เดือนย้อนหลังสำหรับปัญหานี้https://github.com/joyent/node/issues/5698

และปัญหาเป็นเพราะ REPL ล้อมรอบคำสั่งด้วย parens ดังนั้น

foo)(

กลายเป็น

(foo)()

คำอธิบายที่เกิดขึ้นจริงสามารถพบได้ที่นี่https://github.com/joyent/node/issues/5698#issuecomment-19487718


75
และฉันคิดว่าการแทรกเซมิโคลอนอัตโนมัตินั้นไม่ดี
Eric Lippert
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.