目录
一、问题引出
由于JS的构造函数存在内存浪费问题:
javascript
function Star(name,age){
this.name=name
this.age=age
this.sing=function () {
console.log("唱歌!")
}
}
let star01=new Star("Jack",18)
let star02=new Star("Mark",20)
console.log(star01.sing==star02.sing)
打印:不同实例对象的动态方法是不同的,会开出新的内存区域用于存储相同的方法。
data:image/s3,"s3://crabby-images/06c1a/06c1ac92af09f5f2daf0e582d0a0736cd5d92d70" alt=""
对此问题,提出了原型处理方法。
二、prototype原型对象
我们可以将不变的方法直接定义在prototype属性中,以减少内存开销。
data:image/s3,"s3://crabby-images/d3fc9/d3fc94c94b7bdca98458690cd4bd84ad2b9781b9" alt=""
对上面的问题代码进行优化:
javascript
function Star(name,age){
this.name=name
this.age=age
}
Star.prototype.sing=function (){
console.log("唱歌!")
}
let star01=new Star("Jack",18)
let star02=new Star("Mark",20)
console.log(star01.sing==star02.sing)
打印:可以看到两个实例对象的该方法是同一个,证明都使用了同一个原型里的方法,没有新开内存拷贝。
data:image/s3,"s3://crabby-images/3eb74/3eb74a745a43cfc0df8516bdd4021bde597be0ba" alt=""
三、小结
1、公共属性写到构造函数里;
2、公共函数写到prototype原型对象里;
3、构造函数里的this就是指向实例化的对象;
4、原型对象里的this还是指向实例化的对象。
data:image/s3,"s3://crabby-images/7732e/7732ed5e103baccfa99232423b3c8a1af60ad46a" alt=""
四、constructor
prototype里的属性constructor属性就是帮原型确定它的构造函数是谁,如,构造函数Star的prototype里的constructor属性就是指向Star构造函数的。
作用:指向原型对象的构造函数。
data:image/s3,"s3://crabby-images/e65ad/e65ad201bc1a90da29783797eb59e7aef49affd4" alt=""
javascript
function Star(){
}
console.log(Star.prototype)
Star.prototype={
sing:function (){
console.log("唱歌")
},
dance:function (){
console.log("跳舞")
}
}
console.log(Star.prototype)
打印:没构造前,有constructor属性,构造后没有了
可以看到,上面这种写法直接给prototype赋值了新值,没有了constructor对象标识,正确写法应该是:
javascript
function Star() {
}
console.log(Star.prototype)
Star.prototype = {
//重新指回这个原型的构造函数 Star
constructor: Star,
sing: function () {
console.log("唱歌")
},
dance: function () {
console.log("跳舞")
}
}
console.log(Star.prototype)
打印:
data:image/s3,"s3://crabby-images/59367/59367b492d201e50e1932e1e2e3eaa12d5b8629e" alt=""
data:image/s3,"s3://crabby-images/c6405/c6405b744d7e023f20d5bbb0d4775434beb81255" alt=""
五、__proto__对象原型
每个实例化的对象都有一个__proto__,而这个__proto__就是指向构造函数的prototype对象的。这样一来,每个实例化对象都可以访问prototype里的数据了。
data:image/s3,"s3://crabby-images/2d284/2d284ddd56bb0c08629d5325e9e160b923afc1b9" alt=""
1、这是一个只读属性;
2、
data:image/s3,"s3://crabby-images/b17f4/b17f42187e0b0045ed253f19750396d11aa193f9" alt=""
javascript
function Star() {
}
const star01=new Star()
console.log(star01.__proto__ === Star.prototype)
打印:这里可以看到__proto__指向的是构造函数的原型对象
data:image/s3,"s3://crabby-images/c7ecf/c7ecfea8b9b4de15dc33de039499e3e1ffa8f289" alt=""
此外,__proto__实例化对象的原型中也有constructor,并且是指向prototype原型对象中的constructor的。
data:image/s3,"s3://crabby-images/0f0c4/0f0c44ccc20f269a6a8d1e2d87cf0764dac60730" alt=""
data:image/s3,"s3://crabby-images/ac1fc/ac1fc4dd7ac8ef55b850c716287c74b7a038600e" alt=""
六、原型链
只要是对象就是__proto__,prototype对象中也有__proto__属性,我们按照prototype的__proto__可以一级一级的指向,到最顶层Object.prototype.__proto__指向为null。
data:image/s3,"s3://crabby-images/ad66f/ad66f4196e5a47253af34cbae658acf3f245a8cb" alt=""
使用场景:比如某个对象需要调用一个方法,而这个方法在当前对象里没有,就会向上一级查找是否有该方法,若仍然没有再往上寻找prototype里是否有该方法,依次类推,直至到顶层prototype里查找,指向为null时结束查找。
可以通过instanceof来判断对象是否在一个原型链上。
data:image/s3,"s3://crabby-images/9b169/9b16991997a1390545aaa23fb6ebb9f61a42a419" alt=""
data:image/s3,"s3://crabby-images/afbb6/afbb691899732e73a0bb6dcde811828ca723c1de" alt=""