While [] + []
是一个空字符串,[] + {}
is "[object Object]"
和{} + []
is是0
。为什么是{} + {}
NaN?
> {} + {}
NaN
我的问题是,为什么不({} + {}).toString()
为"[object Object][object Object]"
而NaN.toString()
为"NaN"
。
我的问题是为什么仅在客户端发生这种情况?在服务器端(Node.js){} + {}
是"[object Object][object Object]"
。
> {} + {}
'[object Object][object Object]'
总结 :
在客户端:
[] + [] // 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]"
在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)
更新的注释:此问题已在Chrome
49中修复
。
非常有趣的问题!让我们深入。
差异的根源在于Node.js如何评估这些语句与Chrome开发工具如何进行评估。
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);
}
}
);
这就像({}+{})
在Chrome开发人员工具中运行一样,该工具也可以"[object Object][object Object]"
按您期望的方式生成。
另一方面,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;
}
因此,基本上,它call
使用表达式在对象上执行a 。表达式为:
with ((window && window.console && window.console._commandLineAPI) || {}) {
{}+{};// <-- This is your code
}
因此,如您所见,表达式是直接求值的,而没有换行括号。
Node.js的来源证明了这一点:
// This catches '{a : 1}' properly.
Node并不总是这样。这是改变它的实际提交。Ryan对该更改发表了以下评论:“改进了REPL命令的规避方式”,并举例说明了区别。
更新-OP 对 Rhino的 行为方式(以及为何其行为像Chrome devtools和与nodejs不同的行为)感兴趣。
Rhino使用完全不同的JS引擎,这与Chrome开发人员工具和Node.js的REPL都使用V8不同。
这是在Rhino外壳中使用Rhino评估JavaScript命令时发生的事情的基本流程。
外壳运行org.mozilla.javascript.tools.shell.main
。
反过来,例如,如果代码是通过内联开关-e直接传递的,则调用此方法 new IProxy(IProxy.EVAL_INLINE_SCRIPT);
。
这击中了IProxy的run
方法。
它调用evalInlineScript
(src)。这只是编译字符串并评估它。
基本上:
Script script = cx.compileString(scriptText, "<command>", 1, null);
if (script != null) {
script.exec(cx, getShellScope()); // <- just an eval
}
在这三者中,Rhino的外壳是最接近实物的外壳,eval
没有任何包裹。Rhino的eval()
陈述最接近实际陈述,您可以期望其表现完全相同eval
。
问题内容: 我今天注意到,当您输入控制台时,Chrome 49不再输出。而是输出字符串。 为什么是这样?语言改变了吗? 问题答案: 现在,Chromedevtools会自动在隐含的一对括号中包装所有以开头和结尾的内容,以强制将其评估为表达式。这样,现在创建一个空对象。如果您回顾历史记录(),则会看到此内容,前一行将包含在中。 为什么? 我不知道,但是 我可以猜到它减少了对于不了解block-vs-
问题内容: 为什么返回 Javascript? 在文档页面上,我看到以下内容: ## 针对NaN的测试 相等运算符(和)不能用于测试的值。使用代替。 有没有参考资料可以回答这个问题?不客气。 问题答案: 严格的回答 :因为JS规范是这样说的: 如果Type(x)是Number,则 如果x为NaN,则返回false。 如果y为NaN,则返回false。 有用的答案 :浮点数的IEEE 754规范(所
我很熟悉在JavaScript中是“怪异的”,即总是返回,如本文所述。因此,不应进行比较来检查,而应使用isNaN(..)取而代之的是。 所以我惊讶地发现 这似乎不一致。为什么会有这种行为? 它是怎么工作的?方法是否专门检查?
[粤语][1] [1]:https://i.stack.imgur.com/c7n8R.png'C:\用户\busra\下载 C:\用户\busra\下载 C:\用户\busra\下载 C:\用户\busra\下载
问题内容: Google创建了V8 JavaScript引擎:V8在首次执行时将JavaScript源代码直接编译为机器代码。 Node.js基于V8构建-为什么Google不提供像Microsoft Azure这样的Node.js服务器? Google App Engine是放置Node.js的自然之所。 您知道Google为什么不这样做吗? 问题答案: 截至2014年6月,Google在[Go
问题内容: 我正在尝试连接到安全的Web服务。 即使我的密钥库和信任库已正确设置,我也遇到了握手故障。 经过几天的挫败,无休止的谷歌搜索并询问周围的所有人,我发现唯一的问题是java选择在握手期间不将客户端证书发送到服务器。 特别: 服务器请求了客户端证书(CN = RootCA)-即“给我一个由根CA签名的证书” Java查看密钥库,只发现我的客户端证书由“ SubCA”签名,该证书又由“ Ro