首页 关于我们 成功案例 网络营销 电商设计 新闻中心 联系方式
QQ联系
电话联系
手机联系
QQ联系
电话联系
手机联系

JavaScript如何实现继承_有哪些常用方法

发布时间:2025-12-31 00:00
发布者:狼影
浏览次数:
JavaScript继承核心是原型链与构造函数协作,ES6 class extends最推荐,寄生组合式继承经典高效,原型链和借用构造函数各有缺陷。

JavaScript 实现继承的核心思路是让子类能访问父类的属性和方法,同时支持实例化和原型链的正确连接。由于 JS 是基于原型的语言,没有传统 class 的“类继承”机制(ES6 的 class 只是语法糖),所以继承本质上是原型链与构造函数协作的结果。常用方法有 4 种,按现代开发推荐度和兼容性综合来看:ES6 class 继承最简洁可靠,原型链+借用构造函数组合最贴近底层逻辑,而寄生组合式继承是经典高效方案。

ES6 class extends(推荐首选)

这是目前最主流、语义最清晰的方式,底层仍基于原型,但写法接近传统面向对象语言,可读性和维护性高,且天然支持 super、静态方法、getter/setter 等特性。

使用要点:

  • 子类构造函数中必须调用 super(),否则无法访问 this
  • super 既可调用父类构造函数,也可访问父类原型上的方法(如 super.sayHello()
  • 静态方法和 get/set 也会被正常继承

示例:

立即学习“Java免费学习笔记(深入)”;


class Animal {
constructor(name) {
this.name = name;
}
eat() { console.log(this.name + ' 吃东西'); }
static info() { return 'Animal 类'; }
}


class Dog extends Animal {
constructor(name, breed) {
super(name); // 必须先调用
this.breed = breed;
}
bark() { console.log('汪汪!'); }
}

const d = new Dog('旺财', '柴犬');
d.eat(); // 旺财 吃东西
d.bark(); // 汪汪!
Dog.info(); // 'Animal 类'

原型链继承(基础但有缺陷)

通过设置子类原型为父类实例来实现方法复用,简单直接,但所有子类实例共享父类实例的引用类型属性(如数组、对象),且无法向父类构造函数传参。

典型问题:

  • 多个实例修改 this.colors(数组)会互相影响
  • 子类无法在创建时给父类初始化参数

示例:

立即学习“Java免费学习笔记(深入)”;

function Animal(name) {
this.name = name;
this.colors = ['black', 'white'];
}
Animal.prototype.eat = function() { console.log(this.name + ' 吃东西'); };

function Dog() {}
Dog.prototype = new Animal(); // 原型指向父类实例
Dog.prototype.constructor = Dog; // 修复 constructor 指向

const d1 = new Dog();
const d2 = new Dog();
d1.colors.push('brown');
console.log(d2.colors); // ['black', 'white', 'brown'] ← 共享了!

借用构造函数(解决传参与属性隔离)

在子类构造函数中用 callapply 调用父类构造函数,确保每个实例拥有独立的属性副本,也支持传参。但缺点是无法继承父类原型上的方法,代码复用性差。

关键点:

  • 只继承实例属性,不继承原型方法
  • 每次创建子类实例都会执行父类构造函数,可能带来性能开销

示例:

立即学习“Java免费学习笔记(深入)”;

function Animal(name) {
this.name = name;
this.colors = ['black', 'white'];
}
Animal.prototype.eat = function() { console.log(this.name + ' 吃东西'); };

function Dog(name, breed) {
Animal.call(this, name); // 借用构造函数
this.breed = breed;
}

const d = new Dog('小黑', '哈士奇');
console.log(d.name); // '小黑'
console.log(d.breed); // '哈士奇'
d.eat(); // ❌ 报错:d.eat is not a function(没继承原型方法)

寄生组合式继承(经典高效方案)

结合了原型链继承(复用方法)和借用构造函数(隔离属性)的优点,是《JavaScript 高级程序设计》推荐的最优继承模式。它避免了组合继承中父类构造函数被调用两次的问题,只调用一次父类构造函数,并正确设置原型链。

核心步骤:

  • 用一个空函数桥接,把父类原型赋给子类原型(避免执行父类构造函数)
  • 子类构造函数内用 call 借用父类构造函数初始化实例属性
  • 手动修正子类原型的 constructor

封装工具函数(常简写为 inheritPrototype):

function inheritPrototype(subType, superType) {
const prototype = Object.create(superType.prototype); // 创建父类原型副本
prototype.constructor = subType;
subType.prototype = prototype;
}

function Animal(name) { this.name = name; }
Animal.prototype.eat = function() { console.log(this.name + ' 吃东西'); };

function Dog(name, breed) {
Animal.call(this, name); // 第一次也是唯一一次调用 Animal
this.breed = breed;
}
inheritPrototype(Dog, Animal); // 设置原型链
Dog.prototype.bark = function() { console.log('汪!'); };

const d = new Dog('豆豆', '金毛');
d.eat(); // 豆豆 吃东西(继承自原型)
d.bark(); // 汪!(子类自有方法)
console.log(d instanceof Dog); // true
console.log(d instanceof Animal); // true

不复杂但容易忽略:ES6 class 是当前工程实践的默认选择;若需兼容老旧环境(如 IE),寄生组合式继承仍是稳健之选;单纯原型链或借用构造函数适合极简场景或教学理解,但实际项目中应避免单独使用。


# javascript  # es6  # java  # js  # app  # 工具  # 代码复用 


相关文章: C++如何获取当前系统时间?(代码示例)  在 Go 语言中将测试文件移至子目录的正确实践  Microsoft Edge网页按钮无反应怎么办 Microsoft Edge交互修复  Python字符串格式化怎么用_fstring与format全面解析【教程】  简历没回改:利用AI润色让你的文字更专业  《巫师3》次世代新模组发布 致力还原2013年E3演示版光照效果  HTML 中动态设置元素 name 属性的正确写法  Windows10系统怎么查看已安装更新_Win10控制面板卸载补丁  《尘白禁区》发布「猫汐尔-莲驱」角色PV  VSCode的“Go to Symbol”:在文件中快速导航  如何使用切片器在 Excel 中筛选数据  如何在 Vue 中正确设置复选框(Toggle Button)的默认选中状态  ChatGPT 4o 辅助学生复习 GRE 词汇的方法  OPPO Find N5 折叠屏铰链技术升级:更浅折痕与悬停新玩法  如何解决“addFirst(int) 方法未定义”编译错误  如何使用Golang处理时间解析错误_Golangtime.Parse异常捕获与处理  第五人格佣兵致万千无言者怎么获得-第五人格佣兵演绎之星时装介绍  HTML透明颜色代码怎么用HSL调蓝色透明_HSL蓝色透明值设置详解【说明】  Linux运维脚本规范教程_Shell脚本工程化实践  稿定设计AI抠图怎样调整透明度_稿定设计AI透明度滑块与渐变设置【攻略】  Win11右键反应慢怎么办 Win11优化右键菜单加载速度【技巧】  wpsPDF在线编辑入口 wps网页版PDF工具直达  如何在Golang中捕获结构体方法错误_统一返回error类型  Windows10如何更改任务栏高度_Win10解除锁定调整大小  HTML5怎么调整视频播放窗口位置_移动视频在页面中的位置技巧【技巧】  如何使用 Go 正则表达式提取括号内首个纯字母标识符(排除嵌套与后续重复)  在Java中实现简易银行账户管理_Java面向对象实战说明  如何在 Angular 中动态创建可拖拽组件并支持自定义属性配置  如何在 Go 中安全地实现 float32 的原子加法操作  steam网页版登录官方入口_steam官网网页端实时访问教程 


相关栏目: 【 行业资讯17850 】 【 软件资源51899 】 【 网站技术89748 】 【 百度推广44206 】 【 网络营销84187 】 【 运营推广93002 】 【 AI优化91086 】 【 网络优化117696 】 【 网址导航107142