
换言之,只要一个对象没有被刻意切断其原型链——例如通过 Object.create(null) 创建——它就可以直接调用 Object.prototype 这个“公共池”中的基础方法。这究竟意味着什么呢?让我们逐一分析。
继承赋予统一的基础能力
Object.prototype 是绝大多数对象原型链的终点,也是 JavaScript 提供基础行为的默认资料库。这些基础方法包括:
- toString():将对象转换为字符串,例如
{a:1}.toString()的结果是"[object Object]" - hasOwnProperty(prop):检测指定属性是否为对象自身的属性(而非继承自原型链)
- isPrototypeOf(obj):判断当前对象是否作为另一个对象的原型存在
- valueOf() 和 toLocaleString():分别返回对象的原始值以及本地化格式的字符串表示
原型链查找机制如何运作
当你调用 obj.toString() 时,JavaScript 并不会直接在 obj 自身查找该方法,而是沿着原型链逐层搜索:
- 第一步,检查
obj自身是否拥有toString属性 - 如果没有,则继续查找
obj.__proto__(即Object.prototype) - 找到后,执行
Object.prototype.toString
这一查找过程正是原型链查找机制的核心,而 Object.prototype 作为原型链上的最终一环,是最基础、最稳定的公共资源。
构造函数创建的对象同样遵循这一规则
例如,使用 function Person() {} 构造函数创建实例时,new Person() 的 __proto__ 指向 Person.prototype,而 Person.prototype 的 __proto__ 又指向 Object.prototype。因此,即使你未在 Person.prototype 上定义任何方法,实例仍然能够调用 toString() 或 hasOwnProperty()——这一切都归功于原型链的传递机制。
可以覆盖或切断原型链,但需谨慎操作
虽然继承自 Object.prototype 是默认行为,但你完全可以主动干预:
- 使用
Object.create(null)创建“无原型”对象——该对象甚至没有toString方法,调用时会引发错误 - 在实例或原型上定义与
Object.prototype同名的方法(例如toString),会遮蔽原型链上的原始版本 - 直接修改
Object.prototype.toString会影响所有对象——强烈不推荐,这会破坏运行环境的稳定性
因此,默认的继承机制为我们带来了便利,但在修改原型链时,务必充分评估潜在影响。
