文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

javascript中实现继承的示例分析

2023-06-27 10:25

关注

小编给大家分享一下javascript中实现继承的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!

类式继承

//声明父类//声明父类function SuperClass() {  this.superValue = true;}//为父类添加共有方法SuperClass.prototype.getSuperValue = function () {  return this.superValue;};//声明子类function SubClass() {  this.subValue = false;}//继承父类SubClass.prototype = new SuperClass();//为子类添加共有方法SubClass.prototype.getSubValue = function () {  return this.subValue;};var instance = new SubClass();console.log(instance.getSuperValue()); //trueconsole.log(instance.getSubValue()); //false
类式继承需要将父类的实例赋值给子类原型, subClass.prototype 继承了 superClass。这种类式继承有两个缺点。其一,由于子类通过原型 prototype 对父类实例化,继承了父类,父类中的共有属性要是引用类型,就会在子类中被所有实例共用,因此一个子类的实例更改子类原型从父类构造函数中继承来的共有属性就会直接影响到其他子类,如下:
function SuperClass() {    this.courses = ['语文', '数学', '英语']}function SubClass() {}SubClass.prototype = new SuperClass();var instance1 = new SubClass()var instance2 = new SubClass()console.log(instance2.courses) //['语文', '数学', '英语']instance1.courses.push('化学')console.log(instance2.courses) //['语文', '数学', '英语', '化学']
instance1 的修改直接影响了 instance2,这是一个灾难的陷阱。其二,由于子类实现的继承是靠其原型 prototype 对父类的实例化实现的,因此在创建父类的时候,是无法向父类传递参数的,因而在实例化父类的时候也无法对父类构造函数内的属性进行初始化。如何解决这个问题?请继续往下看。

构造函数继承

function SuperClass(current) {  this.courses = ["语文", "数学", "英语"];  this.current = current;}//父类声明原型方法SuperClass.prototype.getCourses= function () {  console.log(this.courses);};//声明子类function SubClass(current) {  SuperClass.call(this, current);}var instance1 = new SubClass("语文");var instance2 = new SubClass("数学");instance1.courses.push('化学')console.log(instance1.courses); //["语文", "数学", "英语", "化学"]console.log(instance1.current); //语文console.log(instance2.courses); //["语文", "数学", "英语"]console.log(instance2.current); //数学instance1.getCourses() //TypeError: instance1.getCourses is not a function
SuperClass.call(this, current) 这条语句是构造函数继承的精华。由于 call 这个方法可以更改函数的作用环境,因此在子类中,对 SuperClass 调用这个 call 就是将子类中变量在父类中执行一遍,由于父类中是给 this 绑定属性的,因此子类也就继承了父类的共有属性。由于这种类型的继承没有涉及原型 prototype,所以父类的的原型方法不会被子类继承,要想被子类继承,只能将 showCourse 放在父类构造函数中,但是这样就违背了代码复用的原则。为了综合以上两种继承的优点,于是有了组合继承。

组合继承

//组合继承function SuperClass(current) {  //引用类型共有属性  this.courses = ["语文", "数学", "英语"];  // 值类型共有属性  this.current = current;}SuperClass.prototype.getCourses = function () {  console.log(this.courses);};SuperClass.prototype.getCurrent = function () {  console.log(this.current);};// 声明子类function SubClass(current, time) {  //构造函数继承父类属性  SuperClass.call(this, current);  this.time = time;}//类式继承 子类原型继承父类SubClass.prototype = new SuperClass();//子类原型方法SubClass.prototype.getTime = function () {  console.log(this.time);};
在子类构造函数中执行父类构造函数,在子类的原型上实例化父类就是组合模式,子类实例中更改父类继承下来的引用类型属性 courses 不会改变其他实例,测试如下
var instance1 = new SubClass("语文", "9:00");instance1.getTime(); //9:00instance1.courses.push('化学')instance1.getCourses(); //["语文", "数学", "英语", "化学"]instance1.getCurrent(); //语文console.log(instance1.current)//语文var instance2 = new SubClass("数学", "10:00");instance2.getTime(); //10:00instance2.getCourses(); //["语文", "数学", "英语"]instance2.getCurrent(); //数学console.log(instance2.current)//数学
但是该模式在执行子类构造函数时执行了一遍父类函数,在实现子类原型继承时又执行了一遍父类构造函数,调用了父类构造函数两遍,显然是一个设计缺陷,那还有更好的方式吗?针对该缺陷,出现了“寄生组合式继承”

寄生组合式继承

在介绍这种继承方式之前,需要先了解 “原型式继承”和“寄生式继承”

基础了解

原型式继承

function inheritObject(o) {  function F() {}  F.prototype = o;  return new F();}var course = {  name: "语文",  alikeCourse: ["数学", "英语"],};var newCourse = inheritObject(course);newCourse.name = "化学";newCourse.alikeCourse.push("物理");var otherCourse = inheritObject(course);otherCourse.name = "政治";otherCourse.alikeCourse.push("历史");console.log(newCourse.name); //化学console.log(newCourse.alikeCourse); //["数学", "英语", "物理", "历史"]console.log(otherCourse.name); //政治console.log(otherCourse.alikeCourse); //["数学", "英语", "物理", "历史"]console.log(course.name); //语文console.log(course.alikeCourse); //["数学", "英语", "物理", "历史"]
inheritObject 可以看做是对类式继承的一种封装,其中的过度类 F 相当于类式继承中的子类。在类式继承中存在的共用引用类型的问题依然存在,但是过度类 F 构造函数中无内容,所以开销较小。

寄生式继承

“寄生式继承”是在“原型式继承”的基础上继续增强。
function inheritObject(o) {  function F() {}  F.prototype = o;  return new F();}var course = {  name: "语文",  alikeCourse: ["数学", "英语"],};function createCourse(obj) {  //通过原型继承方式创建新对象  var o = new inheritObject(obj);  // 拓展新对象  o.getName = function () {    console.log(this.name);  };  return o;}const newCourse = createCourse(course)
这种方式在某个对象内部持续增长属性,像寄生式生长,所以称之为寄生继承。寄生继承是对原型继承的二次封装,并且在二次封装的过程中对继承的对象做了拓展,这样新创建的对象不仅仅有父类中的属性和方法,而且还添加了新的属性和方法。在这种思想的基础上,结合组合式继承,衍生了 “寄生组合式继承”

实现

function inheritObject(o) {    function F() {}    F.prototype = o;    return new F();  }function inheritPrototype(subClass, superClass) {    //复制一份父类的原型副本保存在变量中    var p = inheritObject(superClass.prototype)    //修正因为重写子类原型导致子类的constructor属性被修改    p.constructor = subClass    //设置子类的原型    subClass.prototype = p}
以上父类原型保存一个副本,赋值给子类原型,从而实现继承,且并未重新调用一次父类函数,测试如下,与组合继承模式相近
//testfunction SuperClass(current) {  //引用类型共有属性  this.courses = ["语文", "数学", "英语"];  // 值类型共有属性  this.current = current;}SuperClass.prototype.getCourses = function () {  console.log(this.courses);};SuperClass.prototype.getCurrent = function () {  console.log(this.current);};// 声明子类function SubClass(current, time) {  //构造函数继承父类属性  SuperClass.call(this, current);  this.time = time;}//寄生式继承 子类原型继承父类inheritPrototype(SubClass, SuperClass);//类式继承 子类原型继承父类// SubClass.prototype = new SuperClass();//子类原型方法SubClass.prototype.getTime = function () {  console.log(this.time);};var instance1 = new SubClass("语文", "9:00");var instance2 = new SubClass("数学", "10:00");instance1.getTime(); //9:00instance1.courses.push("化学");instance1.getCourses(); //["语文", "数学", "英语", "化学"]instance1.getCurrent(); //语文console.log(instance1.current); //语文instance2.getTime(); //10:00instance2.getCourses(); //["语文", "数学", "英语"]instance2.getCurrent(); //数学console.log(instance2.current); //数学

区别仅在

//寄生式继承 子类原型继承父类inheritPrototype(SubClass, SuperClass);//类式继承 子类原型继承父类// SubClass.prototype = new SuperClass();
从而实现多个子类多个实例不相互影响,父类构造函数仅仅调用一次,堪当 JavaScript 继承的终极模式。

看完了这篇文章,相信你对“javascript中实现继承的示例分析”有了一定的了解,如果想了解更多相关知识,欢迎关注编程网行业资讯频道,感谢各位的阅读!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯