javascript - 是否有一种与环境无关的方法来检测 Javascript 主机对象?

标签 javascript cross-platform javascript-objects ecma262

我正在编写一个 Javascript 堆栈跟踪库。该库需要检测特定对象或函数是否由程序员创建或是否作为环境的一部分(包括内置对象)存在。宿主对象由于其不可预测的行为而变得有点问题,因此我采用与环境无关的方法来确定 Javascript 中的特定对象是否是宿主对象(请参阅 ECMAScript 3 - 4.3.8)。但是,将宿主对象与 native 对象和原始值区分开来对其他项目中的程序员很有用,尤其是在无浏览器环境中,所以我想关注这一点,而不是宿主对象在我的库中引起的问题或区分程序员-创建的对象。

到目前为止,我只能提出依赖于运行 javascript 代码的环境的解决方案。例如:

// IE Only: does not implement valueOf() in Host Objects
var isHost = (typeof obj === 'object' && typeof obj.valueOf === 'undefined');

// Firefox Only: Host objects have own constructor
var isHost = (obj.constructor && obj.hasOwnProperty('constructor'));

我注意到 jQuery 自己的 isPlainObject()方法也依赖于环境,逻辑比较复杂。

也许这是因为宿主对象就是野兽的本性(因为它们的行为是由环境定义的),但我想进一步挖掘一下,看看这是否可能,并且想知道是否有人遇到过这个特定的问题之前已经准备好了解决方案。

所以。有人知道一个简单的独立于平台的解决方案来测试主机对象吗?如果它在无浏览器环境(如 Node 或 Rhino)中运行,那就更好了。

可能的方法(可能行不通):

  • 测试宿主对象的特征似乎是一个失败的原因,因为没有关于它们行为的规范,但是测试对象是否是 ES3 规范的一部分可能是一种可能性。
  • 我已经尝试使用 Object.prototype.toString() ,因为它的定义非常具体,但结果是不确定的,因为某些环境(即 IE)选择为 native 和主机返回相同的值对象。
  • 可以通过检查原型(prototype)链中对象的最终构造函数是否真的是instanceof Function来做到这一点。

最佳答案

当您查看 definition of host object“由宿主环境提供的对象,以完成 ECMAScript 的执行环境。” — 很明显,没有简单的方法可以确定对象是宿主对象还是 native 对象。

与本地对象不同,宿主对象以特定于实现的方式定义内部属性(例如 [[Prototype]]、[[Class]] 等)。那是因为 specification allows them to do this .但是,宿主对象没有“必须”要求以特定于实现的方式实现内部行为;这是“可能”类型的要求。所以我们不能依赖这个。这些对象可能会或可能不会表现得“怪异”。没办法说。

过去很少尝试检测宿主对象,但所有这些显然都依赖于对特定环境的观察(MSHTML DOM 就是其中之一)——请记住,宿主对象没有任何类型的独特模式/特征来识别。彼得·米修 documented most of the inferences here (看看“功能测试主机对象”部分)。臭名昭著的 typeof ... == "unknown" 来自 MSHTML DOM 及其基于 ActiveX 的宿主对象。请注意,Peter 主要在浏览器脚本上下文中谈论宿主对象,他将检查范围缩小到“这是一个宿主方法吗?”、“这是一个宿主集合对象”等。

在某些环境中,宿主对象不从 Object.prototype 继承(使其易于检查),或者具有引发错误的某些属性(例如,某些“接口(interface)”对象上的“原型(prototype)”在 IE 中),甚至在访问时自己抛出错误。

您似乎可以只检查一个对象是否是规范中定义的对象之一,如果不是,则将其视为宿主。但这并没有什么帮助;它只会给你不是内置的对象。其中一些非标准对象可能仍然是原生的(这意味着它们将实现规范中描述的常用语义)。

您最好的选择是测试您的应用程序/脚本的特定行为,宿主对象可能对其敏感。这始终是最安全的方法。您打算访问某个对象的某些内容吗?从对象中删除一些东西?给对象加点东西?测试它。看看它是否有效。如果没有,您可能正在处理宿主对象。

关于javascript - 是否有一种与环境无关的方法来检测 Javascript 主机对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8645685/

相关文章:

javascript - 访问 javascript 对象中的函数

widget - UserVoice 等托管服务如何将其内容嵌入其他网站?

c++ - 操作系统编程

mercurial - 通过 USB 闪存驱动器使用 Mercurial

javascript - Jquery/JS - 为每个 Div 动态创建对象

Javascript:将对象转换为 Gephi/Sigma.Js 的数组或 JSON

javascript - 在 JS 中测试数组成员资格会产生令人惊讶的结果

javascript - 使用 jQuery 选择元素的一部分?

docker - 根据主机操作系统在Docker Compose中更改镜像

JavaScript - 对象的函数引用其字段