JavaScript进阶(一)— 原型到原型链
在 JavaScript
的世界中,我们常常会通过 构造函数 来创建一个 实例对象:
function Person(name) {
this.name = name
}
const person = new Person('Mark')
console.log(person.name) // Mark
我们使用构造函数Person
,通过 new
创建了一个实例对象 person
。
在实例对象 person
上, 有一个私有属性 __proto__
指向了它的构造函数的原型对象prototype
,
那么,什么是 prototype
, 什么是 __proto__
?
接下来,我们开始进入正题。
prototype
在 JavaScript
中,每个函数都有一个 prototype
属性,这个属性指向了该函数的原型对象。
function Person(name) {
this.name = name
}
Person.prototype.age = 18
const person1 = new Person('Mark')
const person2 = new Person('John')
console.log(person1.name, person1.age) // Mark 18
console.log(person2.name, person2.age) // John 18
可以看到,当 person1
和 person2
均打印 age
属性值为 18
。
这是因为 Person.prototype
正是 实例对象 person1
和 person2
的原型。
既然 prototype
指向的是原型, 那么,原型 又是什么?
简单的理解,在 JavaScript
中,每一个对象在创建时,都有一个与之关联的另一个对象,这个关联的对象就是指原型。
注意
null
是没有原型的。
对象会从其原型对象,继承 属性。这也是为什么 person1
和 person2
均打印 age
属性值为 18
。
__proto__
在 JavaScript
中,每个对象(null
除外)都有一个 私有属性 __proto__
, 这个属性指向了该对象的构造函数的原型prototype
。
function Person(name) {
this.name = name
}
const person = new Person('Mark')
console.log(person.__proto__ === Person.prototype) // true
到这里就会发现,既然构造函数有原型 prototype
,原型也是一个对象,而对象有 __proto__
指向它的构造函数的原型对象, 那么 构造函数的原型对象,是否也有其原型对象呢?
constructor
在说明 原型的原型前,需要了解原型上的一个属性 constructor
, 它指向了原型对象关联的构造函数:
function Person(name) {
this.name = name
}
const person = new Person('Mark')
console.log(person.prototype.constructor === Person) // true
它有助于帮我们理解 找到原型的原型的构造函数。
原型的原型
当我们在控制台打印并输出 Person.prototype.__proto__
时,发现打印了一个对象:
function Person(name) {
this.name = name
}
console.log(Person.prototype.__proto__)
output
:
既然 Person
的原型也有原型, 那么 这个原型的原型对象,它的构造函数又是什么呢?
我们可以使用 constructor
来获取 它的构造函数:
function Person(name) {
this.name = name
}
console.log(Person.prototype.__proto__.constructor)
output
:
可以发现 Person
的原型对象的原型对象,指向的构造函数是 Object
。
即 Person.prototype
的原型,指向的是 Object.prototype
。
那么, Object.prototype
有没有自己的原型呢?
可以发现,Object.prototype
的原型,指向的是 null
。
而null
是表示 没有对象,它没有原型。
原型链
Person.prototype
-> Object.prototype
-> null
(通过 __proto__
进行关联)
这种 对象有一个原型对象,它的原型对象又有它的原型对象,一层层如同链式一样,向上关联,称为 原型链
。
几乎所有 JavaScript
中的对象都是位于原型链顶端的 Object
的实例。
提示
__proto__
是一个非标准的属性,但绝大部分浏览器都支持通过这个属性来访问原型。
__proto__
在实现上是Object
上的一个 getter/setter
访问器 ,使用 obj.__proto__
时,可以理解成返回了 Object.getPrototypeOf(obj)
。
基于原型链的继承
JavaScript
对象是动态的属性“包”(指其自己的属性)。JavaScript
对象有一个指向一个原型对象的链。当试图访问一个对象的属性时,它不仅仅在该对象上搜寻,还会搜寻该对象的原型,以及该对象的原型的原型,依次层层向上搜索,直到找到一个名字匹配的属性或到达原型链的末尾。
function Person(name) {
this.name = name
}
Person.prototype.age = 18
const person1 = new Person('Mark')
const person2 = new Person('John')
console.log(person1.name, person1.age) // Mark 18
console.log(person2.name, person2.age) // John 18
在这个示例中, 虽然Person
的实例对象自身并没有age
属性,但由于它的原型对象上有 age
属性, 实例对象从它的原型对象 继承 了属性 age
。 从而其 age
属性的值为 18
。