原型继承

其他数据类型类似,对象从内置的 Object 原型继承属性和方法,这意味着生成的对象既包含您定义的属性,也包含一个原型属性,其中包含从原型继承的方法

let myObject = {
    'booleanValue' : true
};

myObject;
> Object { booleanValue: true }
    booleanValue: true
    [[prototype]]: Object {  }
            __defineGetter__: function __defineGetter__()
            __defineSetter__: function __defineSetter__()
            __lookupGetter__: function __lookupGetter__()
            __lookupSetter__: function __lookupSetter__()
            __proto__: 
                constructor: function Object()
                hasOwnProperty: function hasOwnProperty()
                isPrototypeOf: function isPrototypeOf()
                propertyIsEnumerable: function propertyIsEnumerable()
                toLocaleString: function toLocaleString()
                toString: function toString()
                valueOf: function valueOf()
                <get __proto__()>: function __proto__()
                <set __proto__()>: function __proto__()

原型属性不应通过属性键直接访问。正如您可能在前一个示例中注意到的那样,浏览器开发者控制台和原型属性键的文档来源中使用的 [[prototype]]<prototype> 表示法暗示了这一点

// Chrome:
let emptyObject = {};

emptyObject;
> {}
  [[prototype]]: Object
// Firefox:
let emptyObject = {};

emptyObject;
> Object {  }
  <prototype>: Object {  }

尽管所有常用浏览器都使用 __proto__ 作为事实标准,但这并未正式标准化,应在生产代码中避免使用。

let emptyObject = {};

emptyObject.__proto__;
> Object {  }
    __defineGetter__: function __defineGetter__()
    __defineSetter__: function __defineSetter__()
    __lookupGetter__: function __lookupGetter__()
    __lookupSetter__: function __lookupSetter__()
    __proto__:
        constructor: function Object()
        hasOwnProperty: function hasOwnProperty()
        isPrototypeOf: function isPrototypeOf()
        propertyIsEnumerable: function propertyIsEnumerable()
        toLocaleString: function toLocaleString()
        toString: function toString()
        valueOf: function valueOf()
        <get __proto__()>: function __proto__()
        <set __proto__()>: function __proto__()

相反,您可以使用内置的 Object.getPrototypeOf()Object.setPrototypeOf() 方法直接访问和修改对象的 [[Prototype]]

let myObj = { "value" : 5 };
let protoParent = { "protoValue" : true };

myObj;
Object { value: 5 }
    value: 5
    <prototype>: Object {  }

Object.getPrototypeOf( myObj );
> Object {  }
    __defineGetter__: function __defineGetter__()
    __defineSetter__: function __defineSetter__()
    __lookupGetter__: function __lookupGetter__()
    __lookupSetter__: function __lookupSetter__()
    __proto__:
    constructor: function Object()
    hasOwnProperty: function hasOwnProperty()
    isPrototypeOf: function isPrototypeOf()
    propertyIsEnumerable: function propertyIsEnumerable()
    toLocaleString: function toLocaleString()
    toString: function toString()
    valueOf: function valueOf()
    <get __proto__()>: function __proto__()
    <set __proto__()>: function __proto__()

Object.setPrototypeOf( myObj, protoParent );
> Object { value: 5 }
    value: 5
    <prototype>: Object { protoValue: true }

为了区分继承属性和作者定义的属性,后者通常称为对象的“自有属性”。

如果指定的属性是对象的直接属性,则内置的 Object.hasOwn() 方法返回 true;如果该属性是继承属性或不存在,则返回 false。只要有可能,就使用 Object.hasOwn() 而不是继承的 hasOwnProperty() 方法,后者不支持 Object.create()

let myObject = {
    'myValue' : 100
};

Object.hasOwn( myObject, 'myValue' );
> true

myObject.__proto__; // The Object prototype inherited by `myObject` is present:
> Object {  }

Object.hasOwn( myObject, '__proto__' ); // The Object prototype inherited by `myObject` is not an "own property:"
> false

检查您的理解程度

为什么应避免使用 __proto__

它未标准化。
许多浏览器不支持它。
它会使您代码的未来维护者感到困惑。