JS继承的实现方式:

既然要实现继承,那么首先我们得有一个父类,代码如下:

function Animal(name) {
// 属性
this.name = name || '小白';
// 实例方法
this.sleep = function () {
console.log(this.name + '正在睡懒觉!');
}
//实例引用属性
this.features = ['11', '22'];
}
// 原型方法
Animal.prototype.eat = function (food) {
console.log(this.name + '正在吃:' + food);
};

1、原型链继承

核心: 将父类的实例作为子类的原型

   function Cat(name) {}
//把Cat的原型指向Animal
Cat.prototype = new Animal();
var tom = new Cat('Tom');
var kissy = new Cat('Kissy'); console.log(tom.name); // "小白"
console.log(kissy.name); // "小白"
console.log(tom.features); // ["11", "22"]
console.log(kissy.features); // ["11", "22"] tom.name = '小黑';
tom.features.push('33'); //针对父类实例值类型成员的更改,不影响
console.log(tom.name); // "小黑"
console.log(kissy.name); // "小白"
//针对父类实例引用类型成员的更改,会通过影响其他子类实例
console.log(tom.features); // ["11", "22", "33"]
console.log(kissy.features); // ["11", "22", "33"]
特点:
  1.非常纯粹的继承关系,实例是子类的实例,也是父类的实例
  2.父类新增原型方法/原型属性,子类都能访问到
  3.简单,易于实现
缺点:
  1.可以在Cat构造函数中,为Cat实例增加实例属性。如果要新增原型属性和方法,则必须放在new Animal()这样的语句之后执行。
  2.来自原型对象的所有属性被所有实例共享(来自原型对象的引用属性是所有实例共享的)(详细请看附录代码: 示例1)
  3.创建子类实例时,无法向父类构造函数传参
 

2、构造继承

核心:使用父类的构造函数来增强子类实例,等于是复制父类的实例属性给子类(没用到原型)

function Cat(name) {
Animal.call(this);
this.name = name || '小黑';
}
var cat = new Cat();
var animl = new Animal();
console.log(cat.name);//小黑
cat.sleep();//小黑正在睡懒觉
console.log(animl.name);//小白
animl.name = '大黄';
console.log(animl.name);//大黄
cat.sleep();//小黑正在睡懒觉!
console.log(cat.name);//小黑
console.log(cat instanceof Animal); // false
console.log(cat instanceof Cat); // true
特点:
  1. 解决了1中,子类实例共享父类引用属性的问题
  2. 创建子类实例时,可以向父类传递参数
  3. 可以实现多继承(call多个父类对象)
缺点:
  1. 实例并不是父类的实例,只是子类的实例
  2. 只能继承父类的实例属性和方法,不能继承原型属性/方法
  3. 无法实现函数复用,每个子类都有父类实例函数的副本,影响性能
 

3、实例继承

核心:为父类实例添加新特性,作为子类实例返回

function Cat(name) {
var instance = new Animal();
instance.name = name || '小黑';
return instance;
}
var cat = new Cat();
console.log(cat.name);//小黑
cat.sleep();//小黑正在睡懒觉!
cat.features.push('33')
console.log(cat.features);//["11", "22", "33"]
console.log(cat instanceof Animal); // true
console.log(cat instanceof Cat); // false
特点:
  1.不限制调用方式,不管是new 子类()还是子类(),返回的对象具有相同的效果
缺点:
  1.实例是父类的实例,不是子类的实例
  2.不支持多继承

4、拷贝继承

核心:通过for循环去拷贝父类的每一个对象

//通过循环去copy父类的每一项
function Cat(name) {
var animal = new Animal();
for (var p in animal) {
Cat.prototype[p] = animal[p];
}
Cat.prototype.name = name || '小黑';
} var cat = new Cat();
console.log(cat.name);//小黑
cat.sleep();//小黑正在睡懒觉!
console.log(cat instanceof Animal); // false
console.log(cat instanceof Cat); // true
特点:
  1.支持多继承
缺点:
  1.效率较低,内存占用高(因为要拷贝父类的属性)
  2.无法获取父类不可枚举的方法(不可枚举方法,不能使用for in 访问到)
 

5、组合继承

核心:通过调用父类构造,继承父类的属性并保留传参的优点,然后通过将父类实例作为子类原型,实现函数复用

function Person(name, age, sex) {
this.name = name;
this.age = age;
this.sex = sex;
}
Person.prototype.sayHi = function () {
console.log("我是Person的方法");
};
function Student(name, age, sex, score) {
//借用构造函数:属性值重复的问题
Person.call(this, name, age, sex);
this.score = score;
}
//改变原型指向----继承
Student.prototype = new Person(); //不传值
//把原型的指向改回原来
Student.prototype.constructor = Student;
Student.prototype.eat = function () {
console.log("我是Student的方法");
};
var stu = new Student("小黑", 20, "男", "100分");
console.log(stu.name, stu.age, stu.sex, stu.score);//小黑 20 男 100分
stu.sayHi();//我是Person的方法
stu.eat();//我是Student的方法
特点:
  1.可以继承实例属性/方法,也可以继承原型属性/方法
  2.既是子类的实例,也是父类的实例
  3.不存在引用属性共享问题
  4.可传参
  5.函数可复用
缺点:
  1.调用了两次父类构造函数,生成了两份实例(子类实例将子类原型上的那份屏蔽了)

js 继承的几种方式的更多相关文章

  1. JS继承的几种方式

    JS作为面向对象的弱类型语言,继承也是其非常强大的特性之一. 既然要实现继承,那么我们先定义一个父类: // 定义一个动物类 function Animal (name) { // 属性 this.n ...

  2. 20. js继承的6种方式

    想要继承,就必须要提供个父类(继承谁,提供继承的属性) 一.原型链继承 重点:让新实例的原型等于父类的实例. 特点: 1.实例可继承的属性有:实例的构造函数的属性,父类构造函数属性,父类原型的属性.( ...

  3. 理解js继承的6种方式

    想要继承,就必须要提供个父类(继承谁,提供继承的属性) 一.原型链继承 重点:让新实例的原型等于父类的实例. 特点:1.实例可继承的属性有:实例的构造函数的属性,父类构造函数属性,父类原型的属性.(新 ...

  4. js继承的6种方式

    想要继承,就必须要提供个父类(继承谁,提供继承的属性) 一.原型链继承 重点:让新实例的原型等于父类的实例. 特点:1.实例可继承的属性有:实例的构造函数的属性,父类构造函数属性,父类原型的属性.(新 ...

  5. js实现继承的5种方式 (笔记)

    js实现继承的5种方式 以下 均为 ES5 的写法: js是门灵活的语言,实现一种功能往往有多种做法,ECMAScript没有明确的继承机制,而是通过模仿实现的,根据js语言的本身的特性,js实现继承 ...

  6. javascript(js)创建对象的模式与继承的几种方式

    1.js创建对象的几种方式 工厂模式 为什么会产生工厂模式,原因是使用同一个接口创建很多对象,会产生大量的重复代码,为了解决这个问题,产生了工厂模式. function createPerson(na ...

  7. js 实现继承的几种方式

    //js中实现继承的几种方式 //实现继承首先要有一个父类,先创造一个动物的父类 function Animal(name){ this.name = name; this.shoot = funct ...

  8. js 实现继承的6种方式(逐渐优化)

    <!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...

  9. JS继承的原理、方式和应用

    概要: 一.继承的原理 二.继承的几种方式 三.继承的应用场景 什么是继承? 继承:子类可以使用父类的所有功能,并且对这些功能进行扩展.继承的过程,就是从一般到特殊的过程.要了解JS继承必须首先要了解 ...

随机推荐

  1. LeetCode 题解之Plus One

    1.题目描述 2.题目分析 从后向前做加法,等于10则进位,否则直接加1 ,返回 digits; 3.代码 vector<int> plusOne(vector<int>&am ...

  2. cookie 常用的几种方法

    { setCookie: function(sName, sValue, oExpires, sPath, sDomain, bSecure) { var sCookie = sName + &quo ...

  3. Vue的路由

    Vue可以实现一种类是ajax不刷新但是切换界面 然后 只是在你的url中的当前地址后面追加信息 首先你要先当如这个路由的模块: <script src="https://unpkg. ...

  4. 铁乐学python_Day42_锁和队列

    铁乐学python_Day42_锁和队列 例:多个线程抢占资源的情况 from threading import Thread import time def work(): global n tem ...

  5. 铁乐学python_day23_面向对象进阶1_反射

    铁乐学python_day23_面向对象进阶1_反射 以下内容大部分摘自博客http://www.cnblogs.com/Eva-J/ isinstance()和issubclass() 两者的返回值 ...

  6. (1)封装 (2)static关键字 (3)继承

    1.封装(重中之重)1.1 基本概念 通常情况下,可以在测试类中给成员变量进行赋值,当给定的数值合法但不合理时,无论是编译还是运行阶段都不会报错或给出提示,此时与现实生活不符. 为了避免上述问题的发生 ...

  7. September 12th 2017 Week 37th Tuesday

    Failure is the fog through which we glimpse triumph. 失败是迷雾,穿过它,我们就可以瞥见光明. Sometimes the fog may be t ...

  8. Maven实战(八)pom.xml简介

    目录 pom作为项目对象模型.通过xml表示maven项目,使用pom.xml来实现.主要描述了项目:包括配置文件.开发者需要遵循的规则.缺陷管理系统.组织和licenses.项目的url.项目的依赖 ...

  9. 【问题定位】tcpdump 抓包简易命令

    tcpdump -A -i lo port 2003 or port 2103 or port 2203 | grep '.*system.*'

  10. 学习日期时间控件 daterangepicker

    aterangepicker 是一款日期时间控件,可选择“年,月,日,时,分,秒”,可选择单面板,也可选择双面板(起止时间). 单面板示例:daterangepicker 单面板 codepen 在线 ...