javascript如何实现继承_ES6的class和原型继承有何不同

发布时间 - 2026-01-11 00:00:00    点击率:
ES6 class本质是原型继承的语法糖,仍通过prototype链和__proto__实现继承;extends自动处理实例与静态继承、强制super()调用、保证new.target正确,而手动模拟易遗漏关键环节。

ES6 class 本质还是原型继承,不是新机制

很多人以为 class 是 JavaScript 新增的“真正面向对象”语法,其实它只是原型继承的语法糖。执行时,class 仍会生成构造函数,并通过 prototype 链和 __proto__ 关系实现继承。你写 class B extends A,底层仍是设置 B.prototype.__proto__ === A.prototype,并用 Object.setPrototypeOf(B, A) 建立静态方法继承。

extends 和手动 Object.setPrototypeOf 的关键区别

手动模拟继承容易漏掉两层关系:实例原型链(__proto__)和构造函数自身继承(constructor 指向、静态方法)。而 extends 自动处理:

  • 设置子类 prototype.__proto__ 指向父类 prototype
  • 设置子类构造函数的 __proto__ 指向父类构造函数(支持静态方法继承)
  • 强制要求子类构造函数中调用 super(),否则 this 不可用
  • 确保 new.target 正确,避免绕过 super() 直接使用 this

手动写等效逻辑需至少三步:

function B() {
  A.call(this); // 继承实例属性
}
Object.setPrototypeOf(B.prototype, A.prototype); // 实例方法
Object.setPrototypeOf(B, A); // 静态方法
但这样仍无法拦截未调用 A.call(this) 的错误,也无法保证 new.target 行为一致。

子类中不调用 super() 会直接报错

这是 class 继承最易踩的坑——只要子类有 constructor,就必须第一行调用 super()。否则运行时报 ReferenceError: Must call super constructor in derived class before accessing 'this' or returning from derived constructor

原因在于:ES6 规范要求子类构造函数中,this 必须由父类构造器初始化(即 super() 内部完成),不能由子类自行创建。而传统函数继承中,你可以跳过 A.call(this),虽然逻辑出错,但不会语法报错。

  • 正确:
    class B extends A {
      constructor(x) {
        super(x); // 必须第一行
        this.x = x;
      }
    }
  • 错误:
    class B extends A {
      constructor(x) {
        this.x = x; // 报错!
      }
    }

静态方法、getOwnPropertyNamesfor...in 行为差异

由于 extends 设置了 B.__proto__ === A,所以静态方法可被继承;但手动用 Object.create(A.prototype) 只影响实例链,不处理静态链。

另外,Object.getOwnPropertyNames(B) 会包含从 A 继承的静态方法(因为 B.__proto__ === A),但 for...in B 不会遍历它们(因静态属性不可枚举)。而原型上的方法在 for...in 中可见,前提是没被设为 enumerable: false

这意味着:用 class 继承时,静态方法是真继承;用旧式 B.__proto__ = A 手动设置也能做到,但多数人只改了 prototype 链,忘了这层。

构造函数内部的 this 初始化时机、静态链完整性、以及对 super() 的强制约束,才是 class 和手写原型继承最实际的分水岭。别被语法迷惑,该查 __proto__ 还得查,该调试 constructor 指向还得调。


# javascript  # es6  # java  # access  # 区别 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何使用Eloquent进行子查询  如何在IIS7中新建站点?详细步骤解析  通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】  如何快速生成橙子建站落地页链接?  js实现获取鼠标当前的位置  Android自定义控件实现温度旋转按钮效果  Angular 表单中正确绑定输入值以确保提交与验证正常工作  北京网站制作公司哪家好一点,北京租房网站有哪些?  如何快速搭建高效WAP手机网站吸引移动用户?  Laravel如何自定义错误页面(404, 500)?(代码示例)  Laravel的Blade指令怎么自定义_创建你自己的Laravel Blade Directives  jQuery中的100个技巧汇总  iOS UIView常见属性方法小结  高端企业智能建站程序:SEO优化与响应式模板定制开发  韩国服务器如何优化跨境访问实现高效连接?  教你用AI润色文章,让你的文字表达更专业  Bootstrap整体框架之JavaScript插件架构  iOS发送验证码倒计时应用  微信小程序 wx.uploadFile无法上传解决办法  胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?  如何挑选高效建站主机与优质域名?  Laravel如何使用withoutEvents方法临时禁用模型事件  php做exe能调用系统命令吗_执行cmd指令实现方式【详解】  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  南京网站制作费用,南京远驱官方网站?  Laravel如何与Docker(Sail)协同开发?(环境搭建教程)  怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?  矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?  Android滚轮选择时间控件使用详解  如何在橙子建站中快速调整背景颜色?  怎么用AI帮你设计一套个性化的手机App图标?  Android利用动画实现背景逐渐变暗  Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理  JavaScript中如何操作剪贴板_ClipboardAPI怎么用  如何在云服务器上快速搭建个人网站?  Laravel如何实现邮箱地址验证功能_Laravel邮件验证流程与配置  如何快速搭建支持数据库操作的智能建站平台?  如何确保西部建站助手FTP传输的安全性?  宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法  Android Socket接口实现即时通讯实例代码  浅谈javascript alert和confirm的美化  Laravel怎么实现支付功能_Laravel集成支付宝微信支付  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】  微信小程序 闭包写法详细介绍  JavaScript如何实现错误处理_try...catch如何捕获异常?  如何破解联通资金短缺导致的基站建设难题?  黑客入侵网站服务器的常见手法有哪些?  简历没回改:利用AI润色让你的文字更专业  EditPlus中的正则表达式 实战(2)