构造函数创建对象
function Person() {
}
var person = new Person();
person.name = 'Kevin';
console.log(person.name) // Kevin
Person 是一个构造函数,我们使用new 创建一个实例对象person
prototype
每个函数都有一个 prototype 属性
function Person() {
}
// 虽然写在注释里,但是你要注意:
// prototype是函数才会有的属性
Person.prototype.name = 'Kevin';
var person1 = new Person();
var person2 = new Person();
console.log(person1.name) // Kevin
console.log(person2.name) // Kevin
那这个函数的 prototype 属性到底指向的是什么呢?是这个函数的原型吗?
函数prototype属性指向了一个对象,这个对象是调用这个构造函数而创建的实例的原型,也就是person1,person2的原型
Person(构造函数) 的 prototype 指向了person.prototype
__ protp __
这是每一个JavaScript对象(除了 null )都具有的一个属性,叫__proto__,这个属性会指向该对象的原型。
- person.__ proto __ === Person.prototype
constructor
一个构造函数可以生成多个实例,但是原型指向构造函数倒是有的,这就要讲到第三个属性:constructor,每个原型都有一个 constructor 属性指向关联的构造函数。
- person.__ proto __ == Person.prototype
- Person.prototype.constructor == Person
- person.constructor===Person.prototype.constructor
当获取 person.constructor 时,其实 person 中并没有 constructor 属性,当不能读取到constructor 属性时,会从 person 的原型也就是 Person.prototype 中读取,正好原型中有该属性
实例和原型
当读取实例的属性时,如果找不到,就会查找与对象关联的原型中的属性,如果还查不到,就去找原型的原型,一直找到最顶层为止。
function Person() {
}
Person.prototype.name = 'Kevin';
var person = new Person();
// 给实例对象 person 添加了 name 属性,当我们打印 person.name 的时候,结果自然为 Daisy
person.name = 'Daisy';
console.log(person.name) // Daisy
// 删除person 的 name 属性时,读取 person.name,从 person 对象中找不到
delete person.name;
// 从 person 的原型也就是 person.__proto__ ,也就是 Person.prototype中查找
console.log(person.name) // Kevin
一些例子
- 当我们创建一个函数时
var person = new Object()
Person 是 Object 的实例,所以 Person 继承了Object 的原型对象Object.prototype上所有的方法, 比如get,set,valueOf,toString
- 创建一个数组时
var num = new Array()
num 是 Array 的实例,所以 num 继承了Array 的原型对象Array.prototype上所有的方法:比如 length,join,pop,push,concat等
但是为什么 var num = [1] 也能继承到很多方法? 因为array.prototype没有,但是他的__ proto __ 里面有
总结
- 所有函数对象的 __ proto __ 都指向 Function.prototype,它是一个空函数(Empty function)
- 所有对象的 __ proto __ 都指向其构造器的 prototype
- 原型和原型链是JS实现继承的一种模型
- 原型链的形成是真正是靠
__proto__而非prototype