为什么{} + {}仅在客户端是NaN?为什么不在Node.js?

Ion*_*zău 135 javascript eval node.js web-developer-toolbar google-chrome-devtools

虽然[] + []是一个空字符串,[] + {}"[object Object]",{} + []0.为什么是{} + {}NaN?

> {} + {}
  NaN
Run Code Online (Sandbox Code Playgroud)

我的问题是不是为什么({} + {}).toString()"[object Object][object Object]"同时NaN.toString()"NaN",这部分在这里有一个答案已经.

我的问题是为什么这只发生在客户端?在服务器端(Node.js){} + {}"[object Object][object Object]".

> {} + {}
'[object Object][object Object]'
Run Code Online (Sandbox Code Playgroud)

总结:

在客户端:

 [] + []              // Returns ""
 [] + {}              // Returns "[object Object]"
 {} + []              // Returns 0
 {} + {}              // Returns NaN

 NaN.toString()       // Returns "NaN"
 ({} + {}).toString() // Returns "[object Object][object Object]"
 var a = {} + {};     // 'a' will be "[object Object][object Object]"
Run Code Online (Sandbox Code Playgroud)

在Node.js中:

 [] + []   // Returns "" (like on the client)
 [] + {}   // Returns "[object Object]" (like on the client)
 {} + []   // Returns "[object Object]" (not like on the client)
 {} + {}   // Returns "[object Object][object Object]" (not like on the client)
Run Code Online (Sandbox Code Playgroud)

Ben*_*aum 130

更新后的注意事项:这已在Chrome 49中修复.

非常有趣的问题!让我们深入研究.

根本原因

差异的根源在于Node.js如何评估这些语句与Chrome开发工具的工作方式.

Node.js做了什么

Node.js使用repl模块.

从Node.js REPL源代码:

self.eval(
    '(' + evalCmd + ')',
    self.context,
    'repl',
    function (e, ret) {
        if (e && !isSyntaxError(e))
            return finish(e);
        if (typeof ret === 'function' && /^[\r\n\s]*function/.test(evalCmd) || e) {
            // Now as statement without parens.
            self.eval(evalCmd, self.context, 'repl', finish);
        }
        else {
            finish(null, ret);
        }
    }
);
Run Code Online (Sandbox Code Playgroud)

这就像({}+{})在Chrome开发人员工具中运行一样,也可以"[object Object][object Object]"按照您的预期生成.

chrome开发人员工具的功能

另一方面,Chrome dveloper工具执行以下操作:

try {
    if (injectCommandLineAPI && inspectedWindow.console) {
        inspectedWindow.console._commandLineAPI = new CommandLineAPI(this._commandLineAPIImpl, isEvalOnCallFrame ? object : null);
        expression = "with ((window && window.console && window.console._commandLineAPI) || {}) {\n" + expression + "\n}";
    }
    var result = evalFunction.call(object, expression);
    if (objectGroup === "console")
        this._lastResult = result;
    return result;
}
finally {
    if (injectCommandLineAPI && inspectedWindow.console)
        delete inspectedWindow.console._commandLineAPI;
}
Run Code Online (Sandbox Code Playgroud)

所以基本上,它call使用表达式对对象执行.表达式为:

with ((window && window.console && window.console._commandLineAPI) || {}) {
    {}+{};// <-- This is your code
}
Run Code Online (Sandbox Code Playgroud)

因此,正如您所看到的,表达式是直接评估的,没有包装括号.

为什么Node.js的行为不同

Node.js的来源证明了这一点:

// This catches '{a : 1}' properly.
Run Code Online (Sandbox Code Playgroud)

Node并不总是这样.这是改变它的实际提交.Ryan在更改中留下了以下评论:"通过差异示例改进REPL命令的方式".


犀牛

更新 - OP对Rhino的行为方式感兴趣(以及为什么它的行为类似于Chrome devtools而且与nodejs不同).

Rhino使用完全不同的JS引擎,不像Chrome开发人员工具和Node.js的REPL都使用V8.

以下是在Rhino shell中使用Rhino评估JavaScript命令时会发生什么的基本管道.

基本上:

Script script = cx.compileString(scriptText, "<command>", 1, null);
if (script != null) {
    script.exec(cx, getShellScope()); // <- just an eval
}
Run Code Online (Sandbox Code Playgroud)

在这三个中,Rhino的外壳是最接近实际eval而没有任何包装的外壳.Rhino是最接近实际eval()陈述的,你可以期望它的表现与预期完全相同eval.

  • @Samuel只需阅读源代码 - 我发誓!在Chrome中,如果输入"调试器"; ,你得到整个管道 - 它将_directly_抛给你'with'只有一个函数上面的`evaluateOn`.在节点中,所有内容都有很好的文档记录 - 它们有一个专用的REPL模块,所有的历史记录都非常好,并且在git上很舒适,之前在我自己的程序中使用了REPL,我知道在哪里看:)我很高兴你喜欢它并找到它它很有用,但我应该熟悉这些代码库(dev-tools和nodejs),而不是我的智力.直接找到源通常总是最容易的. (7认同)
  • +10如果可能的话!哇哇哇哇,......你真的没有生命,或者你比我更聪明地知道这样的事情.请告诉我你搜索了一下才发现这个答案:) (5认同)