Javascript 继承

6 篇文章 0 订阅

    说到继承,先来看一段代码:

var $ = function(selector){
    return new $.prototype.init(selector)
};
$.prototype = {
    constructor:$,
    init:function(selector){
        this.length = selector
    },
    product(val){
        return val * this.length
    },
    length:0
};
$.prototype.init.prototype = $.prototype; //关键
console.log($(6).product(1));  


如果研究过jQuery源码,应该了解,这就是jQuery面向对象的基本思路,关键点在继承,子类继承父类的原型属性和方法。jQuery的巧妙在于:子类同时也是父类的一个原型方法


 1、利用 call、apply 继承

function Food(i){
    this.name = i;
}
Food.prototype.use = "eat";
function Fish(x,y,z){    
    Food.apply(this,[z]);    //Food.call(this,z)     x可省略
    this.color = x;
    this.weight = y    
}
var fish = new Fish("golden","1kg","Salmon");
console.log(fish.name,fish.use);//fish.use:undefined 说明不能继承父类原型的属性和方法

缺点:不能继承父类**原型**的属性和方法

 

 2、原型链继承

function Food(i){
    this.name = i;
}
Food.prototype.use = "eat";
var food = new Food("Cod");
function Fish(x,y){     
    this.color = x;
    this.weight = y 
}
Fish.prototype = new Food("Salmon");  
console.log(Fish.prototype.constructor === Food);//true 此时,构造函数已经指向了Food
Fish.prototype.constructor = Fish;      //因为“Fish.prototype = new Food("Salmon");”将构造函数指向了Food,需要再次指回Fish
var fish = new Fish("golden","1kg");
console.log(fish.name,fish.use);


缺点
· 继承时,要再次生成父类实例,耗费内存;
· 子类要添加属性和方法,只能在“Fish.prototype = new Food("Salmon");”后面。

 

 3、直接继承prototype

 

function Food(i){
    this.name = i
}
function Fish(x,y){        
    this.color = x;
    this.weight = y    
}
Fish.prototype = Food.prototype;    //相当于完全删除了prototype 对象原先的值
console.log(Fish.prototype.constructor === Food);    //true,同样,构造函数已经指向了Food
Fish.prototype.constructor = Fish;    
console.log(Food.prototype.constructor === Fish);    //true true  问题所在:父类和子类的构造函数,都指向了同一个对象:Fish

var fish = new Fish("golden","1kg");
console.log(fish.name,fish.color);//fish.name:undefined 说明不能继承父类构造函数内的属性和方法


缺点:
· 只能继承父类**原型**的属性和方法,不能继承父类构造函数内的属性和方法;
· 父类和子类的构造函数都指向了同一个对象。


 4、 Object.create(父类原型)
Object.create()方法创建一个新对象,使用现有的对象来提供新创建的对象的__proto__。
基于方法3的优化,解决了父类和子类的构造函数指向同一个对象的问题。

function Food(i){
    this.name = i
}
Food.prototype.use = "eat";
function Fish(x,y){    
    this.color = x;
    this.weight = y    
}
Fish.prototype = Object.create(Food.prototype);
Fish.prototype.constructor = Fish;    
console.log(Food.prototype.constructor === Food,Fish.prototype.constructor === Fish);    //true true
var fish = new Fish("golden","1kg","Salmon");
console.log(fish.name,fish.use);                    //undefined "eat"


缺点:只能继承父类原型的属性和方法,不能继承父类构造函数内的属性和方法;

 

5、call/apply + Object.create(父类原型)

 

function Food(i){
    this.name = i
}
Food.prototype.use = "eat";

function Fish(x,y,z){    
    Food.call(this,z);    
    this.color = x;
    this.weight = y    
}
Fish.prototype = Object.create(Food.prototype);
Fish.prototype.constructor = Fish;
var fish = new Fish("golden","1kg","Salmon");
console.log(fish.name,fish.use);


这样,前面的问题,都有效得到解决,算是比较完美的一个方案。

 

 6、ES6 class


ES6中加入了静态方法和属性,也能通过 extends 继承

class Food{
    static staticFn(){
        return "static-fn"
    };
    static staticVal = "static-val";
    constructor(i){
        this.name = i
    };
    info(){
        return "demo-"+this.name
    }

}
class Fish extends Food{    
    constructor(x,y,z){
        super(z);    
        this.color = x;
        this.weight = y    
    }
}
const fish = new Fish("golden","1kg","Salmon");
console.log(fish.name,fish.info());
console.log(Fish.staticFn(),Fish.staticVal);

最后,提前祝各位大小朋友:六一儿童节快乐!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值