新闻中心

j*ascript如何实现继承_有哪些不同的继承方式?

2025-12-13
浏览次数:
返回列表
J*aScript继承有5种方式:1.原型链继承(共享引用属性);2.构造函数继承(私有属性但无原型方法);3.组合继承(功能全但父构造函数调用两次);4.寄生组合继承(只调用一次父构造,ES6底层实现);5.class extends(语法糖,推荐日常使用)。

javascript如何实现继承_有哪些不同的继承方式?

J*aScript 实现继承的核心是让子类能访问并复用父类的属性和方法,同时支持实例化和原型链的正确连接。ES6 之前主要靠手动模拟,ES6 引入 classextends 后语法更简洁,但底层仍基于原型机制。目前主流有 5 种实用继承方式,各有适用场景。

1. 原型链继承

将父类的实例赋给子类的 prototype,子类实例通过原型链向上查找父类方法和共享属性。

优点:简单、复用父类方法;缺点:所有子类实例共享父类引用类型属性(如数组、对象),无法向父类构造函数传参。

function Parent() {
  this.colors = ['red', 'blue'];
}
Parent.prototype.sayHi = function() { console.log('Hi'); };

function Child() {}
Child.prototype = new Parent(); // 关键:继承原型
Child.prototype.constructor = Child; // 修复 constructor

const c1 = new Child();
c1.colors.push('green');
const c2 = new Child();
console.log(c2.colors); // ['red', 'blue', 'green'] ← 被意外修改

2. 构造函数继承(经典借用构造函数)

在子类构造函数中用 callapply 调用父类构造函数,实现属性“私有化”继承。

优点:避免引用类型共享,支持向父类传参;缺点:只能继承实例属性,无法继承原型上的方法(如 sayHi)。

function Parent(name) {
  this.name = name;
  this.colors = ['red', 'blue'];
}

function Child(name, age) {
  Parent.call(this, name); // 关键:借用父类构造逻辑
  this.age = age;
}

const c1 = new Child('Alice', 25);
c1.colors.push('green');
const c2 = new Child('Bob', 30);
console.log(c2.colors); // ['red', 'blue'] ← 独立副本
// c1.sayHi(); // ❌ 报错:没有继承原型方法

3. 组合继承(最常用)

结合前两种:原型链继承方法 + 构造函数继承属性。既复用方法,又保证实例属性独立。

优点:功能完整、语义清晰;缺点:父类构造函数被调用两次(一次在设置原型时,一次在子类中 call)。

Glean Glean

Glean是一个专为企业团队设计的AI搜索和知识发现工具

Glean 210 查看详情 Glean
  • 第一次:Child.prototype = new Parent() → 执行 Parent 构造函数
  • 第二次:Parent.call(this, ...) → 再次执行 Parent 构造函数

4. 寄生组合继承(推荐)

优化组合继承,只调用一次父类构造函数。核心是用 Object.create(Parent.prototype) 创建干净的原型对象,再赋给子类 prototype

这是 ES6 class 的底层实现逻辑,也是 Babel 编译 extends 时采用的方式。

function inheritPrototype(Child, Parent) {
  const prototype = Object.create(Parent.prototype);
  prototype.constructor = Child;
  Child.prototype = prototype;
}

function Parent(name) { this.name = name; }
Parent.prototype.sayHi = function() { console.log('Hi'); };

function Child(name, age) {
  Parent.call(this, name); // 只在这里调用 Parent
  this.age = age;
}
inheritPrototype(Child, Parent); // 关键:干净地继承原型

const c = new Child('Tom', 28);
c.sayHi(); // ✅ 正常输出
console.log(c instanceof Parent); // ✅ true

5. ES6 class extends(语法糖,推荐日常使用)

本质仍是寄生组合继承,但由引擎自动处理,代码更易读、支持 super、静态方法、getter/setter 等。

注意:class 必须用 new 调用,且子类构造函数中必须先调用 super()(否则报错)。

class Parent {
  constructor(name) {
    this.name = name;
  }
  sayHi() { console.log('Hi'); }
}

class Child extends Parent {
  constructor(name, age) {
    super(name); // 必须!等价于 Parent.call(this, name)
    this.age = age;
  }
  sayAge() { console.log(this.age); }
}

const c = new Child('Lisa', 22);
c.sayHi();  // ✅
c.sayAge(); // ✅

基本上就这些。开发中优先用 class extends;需要兼容老环境或深入理解原理时,掌握寄生组合继承就够了。不复杂但容易忽略细节——比如忘记修复 constructor,或 super() 调用时机不对,都会导致行为异常。

以上就是j*ascript如何实现继承_有哪些不同的继承方式?的详细内容,更多请关注其它相关文章!


# 是一个  # 宣城网站百度推广哪家好  # 正规关键词排名包括什么  # seo推广佳选火 星  # 佛山汽车seo方法推广  # 低价网站建设交易  # 避孕套营销推广方案图片  # 洛阳工业品营销推广招聘  # 淄川手机网站优化排行  # 江西seo排名正规公司  # 拉新推广的网站  # 有何不同  # 这是  # javascript  # 如何用  # 报错  # 两次  # 复用  # 有哪些  # 如何实现  # 子类  # red  # app  # java  # es6 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: 网站内容防复制粘贴的实现策略与局限性  Kafka Streams中基于消息头条件过滤消息的实现指南  漫蛙2(台版)官方入口地址 漫蛙2(台版)正版漫画网页端  深入理解字体排版:Adobe光学字偶距与CSS字偶距的差异与实现  QQ邮箱官方网站登录入口_QQ邮箱网页版在线使用  sublime如何配置Go语言开发环境_sublime搭建Golang编译运行系统  汽水音乐在线版入口_汽水音乐网页播放手册  QQ邮箱网页版入口 QQ邮箱官方邮箱登录通道  jQuery Mask 插件中实现电话号码固定前导零的教程  NVIDIA股价11月重挫12%:下月有望好转 但难回5万亿美元巅峰  Golang如何使用buffered channel提高性能_Golang buffered channel优化技巧  解决Tabulator日期时间排序问题的专业指南  QQ邮箱官网登录入口 QQ邮箱网页版邮箱快速登录  mcjs网页版流畅运行 mcjs低配电脑畅玩入口  fishbowl官网免费版 fishbowl养鱼网站入口  CSS图片焦点样式实现教程:理解与应用tabindex属性  c++如何使用折叠表达式(Fold Expressions)_c++17可变参数模板新技巧  如何使用Rector自动化升级旧代码_通过Composer安装和配置Rector进行代码重构  Bilibili动漫最新防封地址发布-Bilibili动漫2025年最稳正版入口推荐  PrimeNG Sidebar背景色自定义指南:CSS覆盖与主题化实践  浏览器打开即用 美图秀秀网页版入口  正确连接J*aScript到HTML实现可点击图片与自定义事件处理  提升Kafka消费者健壮性:会话超时处理与消息处理语义  PS5 Pro有点优势但不多! 《燕云十六声》PS5平台与PC性能画面对比  漫画星球免费下拉式入口 漫画星球免费漫画在线阅读网站  优化 Python 函数中的条件逻辑:解决 if-else 嵌套与参数选择问题  Composer如何解决json扩展缺失的错误  win11如何卸载Windows更新补丁 Win11解决更新导致系统不稳定的问题【修复】  R星幕后开发视频泄露 包含《GTA6》等多款大作  HTML元素状态管理:根据DIV内容动态启用/禁用按钮  Golang如何测试channel通信行为_Golang channel通信测试与分析方法  CSS子选择器:如何区分并样式化嵌套列表的子层级  虫虫漫画精品漫画官网_虫虫漫画精品漫画官网进入精品漫画  实现全屏滚动与导航点:专业教程  漫蛙官网正版漫画入口 漫蛙2官方网页登录地址  想当下一个《2077》?《心之眼》Steam评价升至"多半好评"  Go Martini框架:动态服务解码后的图片内容  Gmail邮箱申请注册直达_Gmail邮箱免费注册PC版官网入口2025  SteamMachine定价或为699美元 大家想入手吗?  深入理解J*a链表中的IPosition接口与使用  vivo云服务网页版登录 怎么登录vivo云服务网页版  怎么在浏览器上运行HTML文件_浏览器运行HTML文件技巧【技巧】  冬*霸灯泡不亮怎么办_浴霸取暖灯一盏不亮的灯座清洁修复法  微信网页版官方快速登录入口 微信网页版网页版账号直达  菜鸟取件码是什么怎么查 最全查询渠道汇总  离线运行Go语言之旅:本地部署与GOPATH配置指南  在J*a中如何开发简易仓库管理与库存统计_仓库管理库存统计项目实战解析  KFC套餐升级怎么获取优惠代码_KFC套餐升级活动与优惠代码获取方法  小红书网页版入口链接分享 小红书官网直接进  在FastAPI中利用lifespan与依赖注入高效管理Redis连接池 

搜索