1、类的声明
2、生成实例
3、如何实现继承
4、继承的几种方式
1、类的声明有哪些方式
<script type="text/javascript">
  //类的声明
  function Animal() {
    this.name = 'name'
  }
  // es6中的class的声明
  class Animal2 {
    constructor() {
      this.name = name
    }
  }
</script>
2、怎么实例化这个对象
console.log(new Animal(), new Animal2());
3、js有几种继承方式
 
1)
/**
* 借助构造函数实现继承
*/
function Parent1() {
  this.name = 'parent1'
}
function Child1() {
  Parent1.call(this);
  this.type = 'child1';
}
console.log(new Child1());
这样就实现了一个继承了。重点看这个Parent1.call,他改变了js运行的上下文,通过这个调用,改变了Parent1的this指向。也就是父类的所有属性都指向了子类的这个实例
缺点:这个构造函数是有自己的原型链的,也就是有自己的prototype属性,虽然说Parent1的属性指向了Child1这个实例,但是他的prototype并没有被Child1所继承。比如说
function Parent1() {
  this.name = 'parent1'
}
Parent1.prototype.say = function(){}; function Child1() {
  Parent1.call(this);
  this.type = 'child1';
}
console.log(new Child1());
看下,是没有say方法的,说明不会继承父类的原型链,所以说,这种继承只是部分继承,如果父类还有一些方法,就拿不到
2)
/**
* 借助原型链实现即成
*/
function Parent2() {
  this.name = "parent2";
}
function Child2() {
  this.type = "child2"
}
Child2.prototype = new Parent2();
console.log(new Child2());
这个方法弥补第一种方式的不足,我们说任何一个构造函数都有一个prototype这个属性的,这个属性的作用是访问他的实例能访问到原型对象上。
如图,我们看到Child2的__proto__指向的是Parent2的原型对象,prototype是Child2的一个属性,这个属性是个对象,将new Parent2()这个对象赋值给了Child2,那么new一个Child2后,访问Child2.name,在Child2里面没有找到,就会去__proto__找,也就是prototype找,也就是Parent2这个实例找,这样就实现了继承
这个继承方式的缺点:在Parent2里面添加,this.play = [1,2,3];,然后new多个Child2的实例,发现他们都指向了同一个play。当s1改变了play,第二个对象跟着被改变了
比如
/**
* 借助原型链实现即成
*/
function Parent2() {
this.name = "parent2";
this.play = [1,2,3]
}
function Child2() {
this.type = "child2"
}
Child2.prototype = new Parent2(); var s1 = new Child2();
var s2 = new Child2();
s1.play.push(4); console.log(s1.play,s2.play);
3)
/**
* 组合方式(结合构造函数 和 原型链的优点)
*/
function Parent3 () {
  this.name = 'parent3';
}
function Child3 () {
  Parent3.call(this);
  this.type = 'child3'
}
Child3.prototype = new Parent3();

我们再加上方法,看看是否能避免上面的问题呢

/**
* 组合方式(结合构造函数 和 原型链的优点)
*/
function Parent3 () {
this.name = 'parent3';
  this.play = [1,2,3];
}
function Child3 () {
  Parent3.call(this);
  this.type = 'child3'
}
Child3.prototype = new Parent3();
var s3 = new Child3();
var s4 = new Child3();
s3.play.push(4);
console.log(s3.play,s4.play);

这个时候发现就不一样了,避免了eg2的缺点,这种组合方式结合了优点,弥补了缺点,这是通常实现继承的方式

缺点:执行了两次父类的原型链,第一次Parent2.call()。第二次 new Parent3()

4)

/**
* 组合继承的优化
*/
function Parent4 () {
  this.name = 'parent4';
  this.play = [1,2,3];
}
function Child4 () {
  Parent4.call(this);
  this.type = 'child4'
}
Child4.prototype = Parent4.prototype;
var s5 = new Child4();
var s6 = new Child4();

这样父类的原型链只执行了一次,但是还剩下一个问题,s5,s6都是父类的实例,没有自己的实例,prototype里面有个contron指明是哪个的实例,而子类的prototype拿的直接是父类的prototype,所以当然拿的是父类的构造函数

/**
* 组合继承的优化2
*/
function Parent5 () {
  this.name = 'parent5';
  this.play = [1,2,3];
}
function Child5 () {
  Parent5.call(this);
  this.type = 'child5'
}
Child5.prototype = Object.create(Parent5.prototype);
Child5.prototype.constructor = Child5;

Object.create方法创建原型对象,Parent5.prototype就是create方法的一个参数,一层一层往上找实现了继承,同时完成了继承,这个就是实现继承的完美方式

js面向对象 继承的更多相关文章

  1. js面向对象继承

    前言 最近看到js面向对象这章节了,主要学习了原型和面向对象继承关系,为了梳理自己的知识逻辑,特此记录. js的面向对象 先说说我目前了解的js创建对象方法 1.写一个函数,然后通过new创建对象 2 ...

  2. 关于JS面向对象继承问题

    1.原型继承(是JS中很常用的一种继承方式) 子类children想要继承父类father中的所有的属性和方法(私有+公有),只需要让children.prototype=new father;即可. ...

  3. js 面向对象 继承

    继承方式有四种: 1.call 2.apply 3.prototype 4.for in call 和 apply 的主要区别: call 传参数只能一个一个的传, apply 因为是用数组,所以可以 ...

  4. JS 面向对象 ~ 继承的7种方式

    前言: 继承 是 OO 语言中的一个最为人津津乐道的概念.许多 OO 语言都支持两种继承方式:接口继承 和 实现继承.接口继承只继承方法签名,而实现继承则继承实际的方法.如前所述,由于函数没有签名,在 ...

  5. js 面向对象 继承机制

    根据w3cschool上的描述:共有3种继承方法(对象冒充,原型链,混合) 1.对象冒充:构造函数ClassA使用this关键字给所有属性和方法赋值,使ClassA构造函数成为ClassB的方法,调用 ...

  6. JS面向对象(3) -- Object类,静态属性,闭包,私有属性, call和apply的使用,继承的三种实现方法

    相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...

  7. JS面向对象(2) -- this的使用,对象之间的赋值,for...in语句,delete使用,成员方法,json对象的使用,prototype的使用,原型继承与原型链

    相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...

  8. js面向对象之继承那点事儿根本就不是事

    继承 说道这个继承,了解object-oriented的朋友都知道,大多oo语言都有两种,一种是接口继承(只继承方法签名):一种是实现继承(继承实际的方法) 奈何js中没有签名,因而只有实现继承,而且 ...

  9. 捋一捋js面向对象的继承问题

    说到面向对象这个破玩意,曾经一度我都处于很懵逼的状态,那么面向对象究竟是什么呢?其实说白了,所谓面向对象,就是基于类这个概念,来实现封装.继承和多态的一种编程思想罢了.今天我们就来说一下这其中继承的问 ...

随机推荐

  1. CI 框架中的日志处理 以及 404异常处理

    最近在整理项目中的日志问题,查了一些关于 “CI 框架中的日志处理 以及 404异常处理” 的东西,顺便记录一下: 关于错误日志: 1. 在CI框架中的 system/core/CodeIgniter ...

  2. Mina初识

    1.概述 1.1 Apache的顶级项目,基于java NIO,支持TCP/IP.UDP/IP: 1.2 Mina对外屏蔽了java NIO使用的复杂性,并在性能上做了不少的优化: 1.3 Mina采 ...

  3. windows 系统 python3.5安装 lxml 库

    有个提示uable find vc***,的错误,如果按照修改python脚本的方法会发现还需要安装VS,安装好了还不一定可以解决问题. 费了半天劲,结合网络上部分信息终于找到了解决方案: 1.打开文 ...

  4. Windows 10 下彻底关闭 Hyper-V 服务

    由于最近需要用到VMWare Workstation 安装虚拟机,安装完成后,发现任何64位的系统都不能正常安装.可能是Hyper-V与VMWare Workstation的冲突造成的不兼容,所以就去 ...

  5. 细说C#中的序列化与反序列化的基本原理和过程

    虽然我们平时都使用第三方库来进行序列化和反序列化,用起来也很方便,但至少得明白序列化与反序列化的基本原理. 懂得人就别看了! 注意:从.NET Framework 2.0 开始,序列化格式化器类Soa ...

  6. Java Bean Validation(参数校验) 最佳实践

    转载来自:http://www.cnblogs.com 参数校验是我们程序开发中必不可少的过程.用户在前端页面上填写表单时,前端js程序会校验参数的合法性,当数据到了后端,为了防止恶意操作,保持程序的 ...

  7. 远程SQL Server连接不上

    运行 cmd -> 输入 netsh winsock reset重启后 应该可以连接sql了

  8. openlayers 聚合效果

    //cyd var cydclusterSource = new ol.source.Cluster({ distance: 40, source: new ol.source.Vector({ fe ...

  9. c#解析json字符串处理(最清晰易懂的方法)

    注:博客迁移到csdn,本文最新地址:https://blog.csdn.net/sajiazaici/article/details/77647625 以下为原文 本文是全网第二简单的方法,因为我女 ...

  10. Python爬虫之图片懒加载技术、selenium和PhantomJS

    一.引入 2.概要 图片懒加载 selenium phantomJs 谷歌无头浏览器 3.回顾 验证码处理流程 一.今日详情 动态数据加载处理 1.图片懒加载 什么是图片懒加载? 案例分析:抓取站长素 ...