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 de。
rived 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; // 报错! } }
静态方法、getOwnPropertyNames 和 for...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)

