JavaScript 基于原型的继承

本贴最后更新于 405 天前,其中的信息可能已经事过景迁

面向对象中的继承特性,有三种实现方案:

JavaScript 采用的是基于原型的实现方案,这也是 JavaScript 最重要的语言特性之一。

ES6 之前,在 JavaScript 中没有类(class)的概念,对象实例是通过构造函数来创建的。构造函数中有个 prototype 属性,该属性的值就是原型。

原型也是一个对象实例,当使用构造函数创建对象实例时,新创建的对象实例就是从这个原型实例“复制”出来的。

每个对象都有一个 __proto__ 属性,当用构造函数创建对象实例时,构造函数的属性 prototype 的值会赋给对象实例的 __proto__ 属性。

空的对象

所有的对象都继承自 Object.prototype,而 Object.prototype 的原型是 null

Object.prototype.__proto__;  // null

Object.prototype 是一个“空”的对象,“空”的对象只有预定义的属性(包括方法)。这些属性会被所有的对象继承,使用下面的语句可以查看这些预定义的属性。

Object.getOwnPropertyNames(Object.prototype).forEach(function log(element, index, array) {
  console.log('[' + index + '] = ' + element);
});
[0] = constructor
[1] = __defineGetter__
[2] = __defineSetter__
[3] = hasOwnProperty
[4] = __lookupGetter__
[5] = __lookupSetter__
[6] = isPrototypeOf
[7] = propertyIsEnumerable
[8] = toString
[9] = valueOf
[10] = __proto__
[11] = toLocaleString

原型链

__proto__ 主要是安放在一个实际的对象中,用它来产生一个原型链,用于寻找方法名或属性,等等。

示例:

var a = {
x: 10,
calculate: function (z) {
return this.x + this.y + z;
}
};

var b = {
y: 20,
__proto__: a
};

var c = {
y: 30,
__proto__: a
};

// call the inherited method
b.calculate(30); // 60
c.calculate(40); // 80

以上示例中的原型链如下图所示:
Aprototypechain.png

注意:ES5 中,规定原型继承需要使用 Object.create() 函数。如下所示:

var b = Object.create(a, {y: {value: 20}});
var c = Object.create(a, {y: {value: 30}});

相关资料

36 | 编程范式游记(7)- 基于原型的编程范式
A prototype chain
《JavaScript 语言精髓与编程实践》

  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    694 引用 • 1142 回帖 • 613 关注
2 操作
lingyundu 在 2020-10-18 22:40:50 更新了该帖
lingyundu 在 2020-10-05 20:09:02 更新了该帖

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...