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

类的声明与实例化
如何实现继承
继承的几种方式
类的声明与实例化
类的声明一般有两种方式

// es5和es6声明类的区别,es5没有统一语法规范。es6有统一写法规范 start。
// es5声明“类”的语法--伪类
// function Person(name,age){//     this.name = name;
//     this.age = age;
//     // this.showName = function(){//     //     alert(this.name);
//     // };
//     // this.showAge = function(){//     //     alert(this.age);
//     // }
// }
// Person.prototype.showName = function(){//     alert(this.name)
// }
// Person.prototype.showAge = function(){//     alert(this.age)
// }
// let p = new Person('blue',18);
// p.showName();
// p.showAge();// es6有单独的声明类的方法
// class Person{//     constructor(name,age){//         this.name = name;
//         this.age = age;
//     }
//     showName(){//         alert(this.name);
//     }
//     showAge(){//         alert(this.age);
//     }
// }
// let p = new Person('red',19)
// p.showName();
// p.showAge();
// es5和es6声明类的区别,es5没有统一语法规范。es6有统一写法规范 end。
// es5和es6的继承区别 ----------------- start
// es5
// function Person(name,age){//     this.name = name;
//     this.age = age;
// }
// Person.prototype.showName = function(){//     alert(this.name)
// }
// Person.prototype.showAge = function(){//     alert(this.age)
// }
// function Worker(name,age,job){//     Person.call(this,name,age);
//     this.job = job;
// }
// Worker.prototype = new Person()
// Worker.prototype.constructor = Worker;
// Worker.prototype.showJob = function(){//     alert(this.job);
// };
// let w = new Worker('huihui',2,'大学教授');
// w.showName();
// w.showAge();
// w.showJob();// es6
class Person{constructor(name,age){this.name = name;this.age = age;}showName(){alert(this.name);}showAge(){alert(this.age);}
}
class Worker extends Person{constructor(name,age,job){super(name,age);this.job = job;}showJob(){alert(this.job);}
}
let w = new Worker('张景辉','28','大学教授');
w.showName();
w.showAge();
w.showJob();
//
// es5和es6的继承区别 ----------------- end

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

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

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

先看个了例子

复制代码```cpp
function Parent1 () {this.name = 'parent1';
}
function Child1 () {Parent1.call(this); //这里的call用apply也可以this.type = 'child1';
}
console.log(new Child1());

复制代码
输出结果

可以看到,生成Child1里面有了父级的属性name,实现了继承。为什么就实现继承了呢?因为在Child1里执行了这句   Parent1.call(this);  如果对this不理解的话,建议看看这个JavaScript作用域和闭包在子类的函数体里执行父级的构造函数,同时改变函数运行的上下文环境(也就是this的指向),使this指向Child1这个类,从而导致了父类的属性都会挂载到子类这个类上去,如此便实现了继承。但这种继承的方法有一个缺点,它只是把父类中的属性继承了,但父类的原型中的属性继承不了。继续上面的代码```cpp
Parent1.prototype.say = function () {console.log("Parent1 prototype")
};new Child1().say()

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

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

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

复制代码
function Parent2 () {this.name = 'parent2';this.play = [1, 2, 3];
}function Child2 () {this.type = 'child2';
}Child2.prototype = new Parent2(); //通过把Child2的原型指向Parent2来实现继承
复制代码
在浏览器中检验一下

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

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

复制代码

var s1 = new Child2();
var s2 = new Child2();
s1.play.push(4);console.log('s1.play:'+s1.play);console.log('s2.play:'+s2.play);

复制代码
打印结果

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

第三种 组合方式

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

看代码

复制代码
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);
复制代码
打印结果

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

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

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

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

第四种 组合方式的优化

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

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

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

复制代码
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();
console.log(s5, s6);

console.log(s5 instanceof Child4, s5 instanceof Parent4);
console.log(s5.constructor);
复制代码

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

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

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

接下来。。。

第五种 组合的完美优化

先看代码吧

复制代码
function Parent5 () {
this.name = ‘parent5’;
this.play = [1, 2, 3];
}

function Child5 () {
Parent5.call(this);
this.type = ‘child5’;
}

//把子类的原型指向通过Object.create创建的中间对象
Child5.prototype = Object.create(Parent5.prototype);

//把Child5的原型的构造函数指向自己
Child5.prototype.constructor = Child5;

//测试
var s7= new Child5();
console.log(s7 instanceof Child5, s7 instanceof Parent5)
console.log(s7.constructor);
复制代码

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

测试结果

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

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

(语法糖)ES6类class声明类的方式 -ES5类声明继承相关推荐

  1. JavaScript 精选:哪些能提高开发效率的es6 新语法糖

    文章目录 一 ECMAScript 相关介绍 1 什么是 ECMA 2 什么是 ECMAScript 3 什么是 ECMA-262 4 谁在维护 ECMA-262 5 为什么要学习 ES6 6 ES6 ...

  2. 【JS(ES6,ES5类的创建及继承)】

    目录 一.ES6的面向对象的语法(ES6创建类) 1.面向过程(了解) 2.面向对象 3.面向对象的特征 4.ECMAScript6(即ES6)的面向对象的语法 5.练习 二.ES6中类的继承 三.E ...

  3. async/await原来这就是语法糖

    带大家基本了解了Promise内部的实现原理,而提到Promise,就不得不提一个东西,那就是async/await,async/await是一个很重要的语法糖,他的作用是用同步方式,执行异步操作.那 ...

  4. Android学习-Kotlin语言入门-变量、函数、语法糖、when、for-in、主构造函数、单例类、函数式API、集合遍历、隐式Intent、Activity生命周期、四种启动模式、标准函数

    探究java语言的运行机制 变量.函数.语法糖 when条件语句 for循环 主构造函数.次构造函数 数据类和单例类 集合的创建与遍历 集合的函数式API 创建菜单 隐式使用Intent Activi ...

  5. scala java抽象理解_Scala学习笔记(五) 抽象类以及类中的一些语法糖

    1. 抽象类 Scala 的抽象类跟Java的一样,不能被实例化. 1.1抽象字段 抽象类中,变量不使用就无需初始化,可以等到子类继承时再进行初始化. scala> abstract class ...

  6. ES6新特性_ES6语法糖class介绍与初体验_就是个语法糖而已---JavaScript_ECMAScript_ES6-ES11新特性工作笔记033

    然后我们看es6中的class,es6中引入这个class这个概念,为了让js的写法更加的 面向对象,写起来方便好理解一些. 其实就是个语法糖而已,他的这个功能,es5的语法都能实现. . 可以看看e ...

  7. 为什么说ES6的class是语法糖?

    0. 前言 我们带着问题去阅读本文: 为什么说ES6的class是语法糖? class是原型的语法糖吗? 那又是如何使用原型来实现class这一语法糖的呢? 1. 基于Prototype的OOP 先来 ...

  8. 了解ES6 The Dope Way第五部分:类,转译ES6代码和更多资源!

    by Mariya Diminsky 通过玛丽亚·迪明斯基(Mariya Diminsky) 了解ES6 The Dope Way第五部分:类,转译ES6代码和更多资源! (Learn ES6 The ...

  9. 语法糖是什么?(ES6的一些小笔记)

    专业的解释 在计算机科学中,语法糖(syntactic sugar)是指编程语言中可以更容易的表达一个操作的语法,它可以使程序员更加容易去使用这门语言:操作可以变得更加清晰.方便,或者更加符合程序员的 ...

最新文章

  1. CPLD/FPGA的UART接口设计之系统时钟(晶振)和波特率关系
  2. TabHost中使用startActivityForResult无法接收返回值的解决方案
  3. scipy minimize当目标函数需要参数、当约束条件特别特别多时
  4. 字节跳动 ClickHouse 在实时场景中的应用和优化实践
  5. lucene 第一天
  6. 计算机无线局域网毕业论文,谈教学设计《计算机网络》网络课程“无线局域网”单元的设计与开发大专毕业论文范文...
  7. 手把手教你用Python求最大值和最小值
  8. 什么是大数据平台和大数据
  9. mac下安装win7虚拟机
  10. 2022年下半年软考考哪个科目?看完就懂了
  11. 网络营销的方案及技巧
  12. ILI9431的LCD屏使用,STM32F1控制
  13. 如何制作出漂亮精致的思维导图?MindNow来帮你
  14. 为什么要放弃Evernote?Effie:为写作而生
  15. matlab MinGW-w64 C/C++ Compiler 的配置(附百度云下载资源)
  16. c语言程序设计需要学多久,九江c语言编程学习,九江学c语言编程报班,九江学c语言编程一般要多久才能学会...
  17. 想看微信读书终生无限卡,瞅这儿
  18. FCC Advanced Algorithm Scripting
  19. 系统设计之图状数据模型
  20. H.266/VVC技术学习:帧内预测之PDPC技术

热门文章

  1. Python的装饰器decorator
  2. SpringBoot集成Swagger2,以及Swagger2常用API
  3. 1121 Damn Single (25分)
  4. 匹兹堡大学计算机科学世界排名,2020年匹兹堡大学QS世界排名
  5. 倍福控制器报1804错误解决方法
  6. 嵌入式必看!全志T113-i+玄铁HiFi4核心板硬件说明资料分享
  7. 学累了之后重新进入学习状态的5种方法,让学习更轻松
  8. Gitment给基于hexo的yilia主题的博客搭建免费评论系统
  9. cors跨域资源共享】同源策略和jsonp
  10. 麦肯锡七步成诗法学习笔记简记以及案例分析(附参考资料可下载)