原型链概念

每个构造函数内部都会有一个(constructor,prototype原型对象),而且都会有一个内置树形__proto__属性用于指向创建它函数对象的prototype原型,当然原型对象也会有__proto__属性,源源不断的指向关联,就形成了一个圆环,形成了原型链

原型prototype对象

函数对象

function f() {
}
console.log(f.prototype)

clipboard.png
我们创建了一个函数f,通过打印它的原型对象,我们会得到一个对象,里面包含一个constructor指向自己本身的指针,还有一个内置__proto__属性

console.log(f.prototype.__proto__)

clipboard.png
那我们再看看它的内置属性指向了哪里,我们可以很清楚的看到constructor指向了Object,而它还内置了toString()和valueOf()方法,这到了哪里已经不用我多说了吧

console.log(f.prototype.__proto__.__proto__)

那我们再继续寻找往下找呢,我们会得到null,此可以得出结论,请看下面图解

clipboard.png

普通对象

let obj = {}
console.log(obj.prototype) // undefined

当然普通对象是没有原型对象的

作用

原型链就是为了实现继承

例子

function Father() {
    this.name = 'aaa'
}

Father.prototype.f1 = function () {
    alert(this.name)
}

// 子函数 继承父
function Son(name) {
    Father.call(this, name)
}

// 我认为继承父
Son.prototype = Object.create(Father.prototype)

// minix 继承多种 Object.assign(1,2,3)
Son.prototype.constructer = Son   //构造器指向

// 重写父方法,改变数据变化,并且不会覆盖父方法,这里称为多态
Son.prototype.f1 = function () {
    console.log(this.name)
}

var S1 = new Son()
S1.f1() // 调用 aaa

检查原型和实例的关系

第一种使用 instanceof 操作符
    console.log(S1 instanceof Object)//true
    console.log(S1 instanceof Father)//true
    console.log(S1 instanceof Son)//true
    
上面检测S1这个实例是否属于Object,Father,Son,结果都返回了true,有些小伙伴会疑问为什么我new的是Son,它还会属于Father,因为Son.prototype = Object.create(Father.prototype)这段代码已经将父的原型给了子了,并且又constructer指向了子函数,这相当于已经继承成功




第二种使用isPrototypeOf

作用:检测一个对象是否是另一个对象的原型。或者说一个对象是否被包含在另一个对象的原型链中

    console.log(Object.prototype.isPrototypeOf(S1))  //true
    console.log(Father.prototype.isPrototypeOf(S1))  //true
    console.log(Son.prototype.isPrototypeOf(S1))   //true
    

查找

Function.__proto__ === Function.prototype

每个函数声明之后都会生成一个函数对象,这正如我们说的那样函数的__proto__是指向它自己的原型的

Function.prototype.__proto__  === Object.prototype

函数原型再向上查找是什么,它们直接通过什么连接,当然是__proto__,它指向的则是Object.prototype

Object.prototype.__proto__  === null

然后再去查找Object.prototype,它会等于null,因为已经到达最底层了,这个记住就好

constructor属性

constructor属性不影响任何JavaScript的内部属性。
instanceof检测对象的原型链,通常你是无法修改的

constructor其实没有什么用处,只是JavaScript语言设计的历史遗留物。
由于constructor属性是可以变更的,所以未必真的指向对象的构造函数,
只是一个提示。不过,从编程习惯上,我们应该尽量让对象的constructor指向其构造函数,以维持这个惯例
总结
原型和原型链是JS实现继承的一种模型
原型链是靠__proto__ 形成的,它在其中的作用属于连接的线

me_zhazha
291 声望6 粉丝