本文是学习中传思客在慕课网开的课程《前端跳槽面试必备技巧》的学习笔记。课程地址:https://coding.imooc.com/class/evaluation/129.html#Anchor

本文将从以下几方面介绍类与继承

  • 类的声明与实例化
  • 如何实现继承
  • 继承的几种方式

类的声明与实例化

类的声明一般有两种方式

  1. //类的声明
  2. var Animal = function () {
  3. this.name = 'Animal';
  4. };
  5.  
  6. //ES6中类的声明
  7. class Animal2 {
  8. constructor () {
  9. this.name = 'Animal2';
  10. }
  11. }

实例化就比较简单,直接用new运算符

  1. new Animall()
  2.  
  3. new Animal2()

这些比较简单,简单介绍一下就可以了。接下来,介绍本文的重点内容,继承。

如何实现继承

实现继承的方式主要有两种:

第一种借助构造函数实现继承

先看个了例子

  1. function Parent1 () {
  2. this.name = 'parent1';
  3. }
  4. function Child1 () {
  5. Parent1.call(this); //这里的call用apply也可以
  6. this.type = 'child1';
  7. }
  8. console.log(new Child1());

输出结果

可以看到,生成Child1里面有了父级的属性name,实现了继承。为什么就实现继承了呢?

因为在Child1里执行了这句   Parent1.call(this);  如果对this不理解的话,建议看看这个JavaScript作用域和闭包

在子类的函数体里执行父级的构造函数,同时改变函数运行的上下文环境(也就是this的指向),使this指向Child1这个类,从而导致了父类的属性都会挂载到子类这个类上去,如此便实现了继承。

但这种继承的方法有一个缺点,它只是把父类中的属性继承了,但父类的原型中的属性继承不了。继续上面的代码

  1. Parent1.prototype.say = function () {
  2. console.log("Parent1 prototype")
  3. };
  4.  
  5. new Child1().say()

从结果中可以看出 Child1中是没有say方法的,因为say是加在父类的原型上的,这种继承方式只改变父类构造函数在子类函数体中的指向,继承不了原型的属性。

第二种是借助原型链实现继承

原型链这里直接用了,不再详细介绍了,如果对原型链还不是很了解的话,建议先看看这个,详谈Javascript原型链

  1. function Parent2 () {
  2. this.name = 'parent2';
  3. this.play = [1, 2, 3];
  4. }
  5.  
  6. function Child2 () {
  7. this.type = 'child2';
  8. }
  9.  
  10. Child2.prototype = new Parent2(); //通过把Child2的原型指向Parent2来实现继承

在浏览器中检验一下

可以看到在Child2的实例的__proto__的属性中有Parent2的属性,由此实现了Child2从Parent2的继承。

但这种继承方式也有不足。接着看代码

  1. var s1 = new Child2();
  2. var s2 = new Child2();
  3. s1.play.push(4);

console.log('s1.play:'+s1.play);
  console.log('s2.play:'+s2.play);

打印结果

我们只改了s1这个实例的属性,却发现Child2的其他实例的属性都一起改变了,因为s1修改的是它原型的属性,原型的属性修改,所有继承自该原型的类的属性都会一起改变,因此Child2的实例之间并没有隔离开来,这显然不是我们想要的。

第三种 组合方式

组合方式就是前两种方法组合而成的,上面两种方式都有不足,这种方式就解决了上面两种方式的不足。

看代码

  1. function Parent3 () {
  2. this.name = 'parent3';
  3. this.play = [1, 2, 3];
  4. }
  5.  
  6. function Child3 () {
  7. Parent3.call(this); //子类里执行父类构造函数
  8. this.type = 'child3';
  9. }
  10.  
  11. Child3.prototype = new Parent3(); //子类的原型指向父类
  12.  
  13. //以下是测试代码
  14. var s3 = new Child3();
  15. var s4 = new Child3();
  16.  
  17. s3.play.push(4);
  18.  
  19. console.log(s3.play, s4.play);

打印结果

可以看出,修改某个实例的属性,并不会引起父类的属性的变化。

这种方式的继承把构造函数和原型链的继承的方式的优点结合起来,并弥补了二者的不足,功能上已经没有缺点了。

但这种方法仍不完美,因为创建一个子类的实例的时候,父类的构造函数执行了两次。

每一次创建实例,都会执行两次构造函数这是没有必要的,因为在继承构造函数的时侯,也就是Parent3.call(this)的时候,parnet的属性已经在child里运行了,外面原型链继承的时候就没有必要再执行一次了。所以,接下来我们对这一方法再做一个优化。

第四种 组合方式的优化

上面一种继承方式问题出在继承原型的时候又一次执行了父类的构造函数,所以优化就从这一点出发。

组合方式中为了解决借助构造函数继承(也就是本文中第一种)的缺点,父类的原型中的属性继承不了,所以才把子类的原型指向了父类。

但是父类的属性,在子类已经中已经存在了,子类只是缺少父类的原型中的属性,所以,根据这一点,我们做出优化。

  1. function Parent4 () {
  2. this.name = 'parent4';
  3. this.play = [1, 2, 3];
  4. }
  5.  
  6. function Child4 () {
  7. Parent4.call(this);
  8. this.type = 'child4';
  9. }
  10.  
  11. Child4.prototype = Parent4.prototype; //优化的点在这里
  12.  
  13. //以下为测试代码
  14. var s5 = new Child4();
  15. var s6 = new Child4();
  16. console.log(s5, s6);
  17.  
  18. console.log(s5 instanceof Child4, s5 instanceof Parent4);
  19. console.log(s5.constructor);

在这种继承方式中,并没有把直接把子类的原型指向父类,而是指向了父类的原型。这样就避免了父类构造函数的二次执行,从而完成了针对组合方式的优化。但还是有一点小问题,先看输出结果

可以看到s5是new Child4()出来的,但是他的constructor却是Parent4.

这是因为Child4这个类中并没有构造函数,它的构造函数是从原型链中的上一级拿过来的,也就是Parent4。所以说到这里,终于能把最完美的继承方式接受给大家啦。

接下来。。。

第五种 组合的完美优化

先看代码吧

  1. function Parent5 () {
  2. this.name = 'parent5';
  3. this.play = [1, 2, 3];
  4. }
  5.  
  6. function Child5 () {
  7. Parent5.call(this);
  8. this.type = 'child5';
  9. }
  10.  
  11. //把子类的原型指向通过Object.create创建的中间对象
  12. Child5.prototype = Object.create(Parent5.prototype);
  13.  
  14. //把Child5的原型的构造函数指向自己
  15. Child5.prototype.constructor = Child5;
  16.  
  17. //测试
  18. var s7= new Child5();
  19. console.log(s7 instanceof Child5, s7 instanceof Parent5)
  20. console.log(s7.constructor);

本例中通过把子类的原型指向Object.create(Parent5.prototype),实现了子类和父类构造函数的分离,但是这时子类中还是没有自己的构造函数,所以紧接着又设置了子类的构造函数,由此实现了完美的组合继承。

测试结果

总结:

本文并没有直接把最完美的继承直接写出来,而是由浅入深循序渐进的来介绍的,如果对后面几种方法没看太懂的话,可能是原型链掌握的不够好,还是建议看看这个详谈Javascript原型链

类的继承就告一段落了,这部分内容确实不好理解,文章写起来也不好写,可能有的地方语言组织的也不好,有点难懂。大家凑合着看,多看几遍,敲一敲代码就懂了。

如果觉得本文对你有帮助就点个赞吧^_^

详谈Javascript类与继承的更多相关文章

  1. JavaScript 类式继承与原型继承

    交叉着写Java和Javascript都有2年多了,今天来总结下自己所了解的Javascript类与继承. Javascript本身没有类似Java的面向对象的类与继承术语,但其基于原型对象的思想却可 ...

  2. javascript类式继承最优版

    直接看实例代码: <!doctype html> <html lang="en"> <head> <meta charset=" ...

  3. JavaScript “类”定义 继承 闭包 封装

    一.Javascript “类”: 类:在面向对象编程中,类(class)是对象(object)的模板,定义了同一组对象(又称"实例")共有的属性和方法. Javascript是一 ...

  4. javascript类式继承模式#4——共享原型

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  5. javascript类式继承模式#3——借用和设置原型

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  6. javascript类式继承模式#2——借用构造函数

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  7. javascript类式继承模式#1——默认模式

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  8. javascript类式继承函数最优版

    直接上代码: klass函数 var klass = function (Parent, props) { var Child, F, i; //1.新构造函数 Child = function () ...

  9. javascript“类”与继承总结和回顾

    Javascipt语法不支持"类"(class)[es6已经支持],但是有模拟类的方法.今天我主要谈谈Javascipt中模拟“类”的方法及js中继承的总结和回顾. js中实现“类 ...

随机推荐

  1. NodeJs通过镜像下载相关NPM模块

    临时通过镜像使用一次:npm --registry https://registry.npm.taobao.org install  模块名[设置镜像源地址为淘宝] 持久使用的第一种方法: npm c ...

  2. Tomcat 笔记-设置虚拟主机

    通过作用虚拟主机,可以使多个不同域名的网站共存于一个Tomcat中 在tomcat的server.xml文件中添加主机名: <Host name="hostname" app ...

  3. Nginx监控-Nginx+Telegraf+Influxb+Grafana

    搭建了Nginx集群后,需要继续深入研究的就是日常Nginx监控. Nginx如何监控?相信百度就可以找到:nginx-status 通过Nginx-status,实时获取到Nginx监控数据后,如何 ...

  4. Mysql 用户,权限管理的几点理解。

    前两天项目数据库要移植到mysql,为此临时抓了几天很久没用的mysql. 公司的数据库比较简单,从oracle迁移到mysql很简单,但是,中间的权限管理让我感觉既简单又复杂..简单是因为网上关于m ...

  5. Appium python自动化测试系列之等待函数如何进行实战(九)

    ​9.1 等待函数的使用 9.1.1 为什么要使用等待函数 我们在做自动化的时候很多时候都不是很顺利,不是因为app的问题,我们的脚本也没问题,但是很多时候都会报错,比如一个页面本来就有id为1的这个 ...

  6. Appium python自动化测试系列之自动化截图(十一)

    11.1 截图函数的正常使用 11.1.1 截图方法 无论是在手动测试还是自动化测试中场景复现永远是一个很重要的事情,有时候一些问题可能很难复现,这个都需要测试人员对bug有很高的敏感度,在一般的情况 ...

  7. Linux_服务器_01_查看公网IP

    在linux终端提示符下,输入以下命令: 精选: curl icanhazip.com/curl ifconfig.mecurl ipecho.net/plain 可以看到下图已经查询到公网IP地址了 ...

  8. x86-64栈帧中的“红色区域” red zone of stack frame on x86-64

    前几天看System V AMD64 ABI标准的时候发现栈帧的顶部后面有一块"red zone",在学cs:app3e/深入理解操作系统的时候并没有遇到这个,总结一下. 引用标准 ...

  9. 349B - C. Mafia

    C - Mafia Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u Submit S ...

  10. js中的事件缓存机制

    异步任务指的是,不进入主线程.而进入"任务队列"(task queue)的任务,只有"任务队列"通知主线程,某个异步任务可以执行了,该任务才会进入主线程执行. ...