JavaScript中的类定义和继承实现

ES5中因为没有class关键字,所以创建类的方式是通过构造函数来定义的。我将一步步的用代码演示如何慢慢用原生的语法实现JS的类的定义和继承。希望大家喜欢。

废话不多说,我们来看原生JavaScript定义类的方法。

1. 最简单的类

// 类的构造函数
function Person() {
    this.name = "夏诺风"; // 属性 name
    this.age = 18; // 属性 age
}

// 调用类则使用new关键字
var person = new Person();
console.log(person.name); // 输出 夏诺风


2. 构造函数和原型链里面增加属性和方法

function Person() {
    this.name = "夏诺风"; // 属性 name
    this.age = 18; // 属性 age

    // 增加类方法
    this.run = function() {
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function() { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

// 调用类则使用new关键字
var person = new Person();
console.log(person.name); // 输出 "夏诺风"
person.run(); // 调用方法 输出 "夏诺风正在跑步"

// 我们也可以调用原型链的方法
person.walk(); // 调用原型链上的方法 输出 "夏诺风正在走路"

以上两个例子就是ES5中怎么定义类的属性和方法的例子,你可以在构造函数中定义也可以在原型链上定义,都是没有问题的。

3. 类里面的静态方法

function Person() {
    this.name = "夏诺风"; // 属性 name
    this.age = 18; // 属性 age

    // 增加类方法
    this.run = function() { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 静态方法
Person.getInfo = function() {
    console.log('我是静态方法');
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function() { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

// 静态方法的调用是不需要new的,而是直接“构造方法名称.方法名”
Person.getInfo();

ES5里面的继承

1. 对象冒充实现继承

function Person() {
    this.name = "夏诺风"; // 属性 name
    this.age = 18; // 属性 age

    // 增加类方法
    this.run = function() { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function() { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

// SuperMan类继承Person 在ES5中继承最常用的方法就是原型链+对象冒充的组合继承模式
function SuperMan() {
    Person.call(this); // 对象冒充实现继承
}

var superman = new SuperMan();
// 对象冒充可以继承构造函数中的属性和方法,但是没有办法结成原型链上的属性和方法。
superman.run(); // 输出 “夏诺风正在跑步”
superman.walk(); // 报错,对象冒充实现的继承没有办法继承原型链上的属性和方法。

2. 原型链实现继承

function Person() {
    this.name = "夏诺风"; // 属性 name
    this.age = 18; // 属性 age

    // 增加类方法
    this.run = function() { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function() { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

// SuperMan类继承Person 在ES5中继承最常用的方法就是原型链+对象冒充的组合继承模式
function SuperMan() {}

SuperMan.prototype = new Person(); // 原型链实现继承

var superman = new SuperMan();
// 原型链实现继承,既可以继承构造函数中的属性和方法,也可以继承原型链上的函数和方法。
superman.run(); // 输出 “夏诺风正在跑步”
superman.walk(); // 输出 “夏诺风正在走路”

3. 原型链继承的问题?没有办法给父类传参

function Person(name, age) {
    this.name = name; // 属性 name
    this.age = age; // 属性 age

    // 增加类方法
    this.run = function () { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function () { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

function SuperMan(name, age) {}

SuperMan.prototype = new Person();

var superman = new SuperMan("超级夏诺风", 19); // 原型链继承的时候,我们没有办法给父类传参数
superman.run(); // 输出 “undefined正在跑步”

4. 原型链 + 对象冒充继承的组合继承模式

function Person(name, age) {
    this.name = name; // 属性 name
    this.age = age; // 属性 age

    // 增加类方法
    this.run = function () { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function () { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

function SuperMan(name, age) {
    Person.call(this, name, age); // 对象冒充继承 实例化子类可以给父类传参
}

SuperMan.prototype = new Person();

var superman = new SuperMan("超级夏诺风", 19); // 原型链继承的时候,我们没有办法给父类传参数
superman.run(); // 输出 “超级夏诺风正在跑步”

5. 原型链 + 对象冒充继承的组合继承模式的另外一种方法

function Person(name, age) {
    this.name = name; // 属性 name
    this.age = age; // 属性 age

    // 增加类方法
    this.run = function () { // 实例方法必须通过new以后才能被调用
        console.log(this.name + '正在跑步');
    }
}

// 原型链上的属性会被多个实例所共享,构造函数中的属性则不会
Person.prototype.sex = "男"; // 原型链增加属性

Person.prototype.walk = function () { // 原型链上增加方法
    console.log(this.name + '正在走路');
};

function SuperMan(name, age) {
    Person.call(this, name, age); // 对象冒充继承 实例化子类可以给父类传参
}

SuperMan.prototype = Person.prototype; // 因为上面已经继承了父类的构造函数中的属性和方法,剩下我们只需要继承原型链就好了

var superman = new SuperMan("超级夏诺风", 19); // 原型链继承的时候,我们没有办法给父类传参数
superman.run(); // 输出 “超级夏诺风正在跑步”

基本涵盖了ES5中的类创建和类继承的实现方式,初次总结希望大家喜欢,有不足之处也希望大家指出。

转载于:https://my.oschina.net/alexskywinner/blog/3073116

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JavaScript 实现继承方法有以下几种方式: 1. 原型继承:利用原型链实现继承。通过将父的实例作为子的原型,子就可以访问到父的属性和方法。例如: ```javascript function Parent() { this.name = 'parent'; } Parent.prototype.sayName = function() { console.log(this.name); } function Child() {} Child.prototype = new Parent(); var child = new Child(); child.sayName(); // 输出:parent ``` 2. 构造函数继承:利用 call 或 apply 方法将父构造函数的作用域赋给子。这种方式可以实现继承。例如: ```javascript function Parent(name) { this.name = name; } function Child(name) { Parent.call(this, name); } var child = new Child('child'); console.log(child.name); // 输出:child ``` 3. 组合继承:结合原型继承和构造函数继承,既可以继承的属性和方法,又可以实现实例的独立。例如: ```javascript function Parent(name) { this.name = name; } Parent.prototype.sayName = function() { console.log(this.name); } function Child(name, age) { Parent.call(this, name); this.age = age; } Child.prototype = new Parent(); Child.prototype.constructor = Child; var child = new Child('child', 18); console.log(child.name); // 输出:child console.log(child.age); // 输出:18 child.sayName(); // 输出:child ``` 4. 原型式继承:利用 Object.create() 方法创建一个新对象,以某个对象为原型,然后再对新对象进行修改。例如: ```javascript var parent = { name: 'parent', sayName: function() { console.log(this.name); } }; var child = Object.create(parent, { name: { value: 'child' } }); child.sayName(); // 输出:child ``` 5. 寄生式继承:与原型式继承似,但是在增强对象的方法时使用了一个函数封装。例如: ```javascript var parent = { name: 'parent', sayName: function() { console.log(this.name); } }; function createChild(original) { var child = Object.create(original); child.sayName = function() { console.log('hello, ' + this.name); } return child; } var child = createChild(parent); child.sayName(); // 输出:hello, parent ``` 需要注意的是,在实现继承时应该注意避免出现属性和方法的重复定义,以及避免在父的原型上修改引用型的属性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值