什么是javascript混入_它如何实现多继承效果?
发布时间 - 2026-01-07 00:00:00 点击率:次JavaScript混入(Mixin)非语言原生特性,是通过Object.assign或高阶类工厂模拟多继承的复用模式,不改变继承链,但需注意同名方法覆盖、不可枚举属性遗漏及副作用管理。
JavaScript 混入(Mixin)不是语言原生特性
JavaScript 本身不支持传统面向对象语言中的多继承,class 只允许单继承(extends 后只能跟一个父类)。混入是开发者为模拟“多个类共享行为”而总结出的模式,本质是把一组方法复制到目标对象或原型上。它不改变继承链,也不创建新类关系,只是复用函数逻辑。
最常用实现:Object.assign + 函数式 mixin 工厂
典型做法是写一个接收目标对象和源对象的函数,用 Object.assign 把源对象的方法拷贝过去。这种方式轻量、可组合、不污染全局。
function MixinLogger(target) {
target.log = function(msg) { console.log(`[LOG] ${msg}`); };
return target;
}
function MixinTimer(target) {
target.startTimer = function() { this.startTime = Date.now(); };
target.elapsed = function() { return Date.now() - this.startTime; };
return target;
}
class Task {}
MixinLogger(Task.prototype);
MixinTimer(Task.prototype);
const t = new Task();
t.startTimer();
console.log(t.elapsed()); // 输出毫秒数
-
MixinLogger和MixinTimer是独立函数,互不影响,可按需调用 - 必须显式传入
Task.prototype,否则方法不会出现在实例上 - 若多个 mixin 定义同名方法(如都加了
init),后执行的会覆盖先执行的——没有自动合并策略
ES6 class 场景下用高阶类工厂模拟“多继承”
想在 class 语法中“继承多个行为”,可用函数返回类的模式,把 mixin 封装成可叠加的类工厂。
const WithLogger = (BaseClass) => class extends BaseClass {
log(msg) { console.log(`[LOG] ${msg}
`); }
};
const WithTimer = (BaseClass) => class extends BaseClass {
startTimer() { this.startTime = Date.now(); }
elapsed() { return Date.now() - this.startTime; }
};
class Task {}
const TaskWithFeatures = WithTimer(WithLogger(Task));
const t = new TaskWithFeatures();
t.log('task started');
t.startTimer();
- 执行顺序很重要:
WithTimer(WithLogger(Task))表示先混入Logger,再在其基础上混入Timer - 所有 mixin 类工厂都必须接受并返回一个类,否则
extends会报Uncaught TypeError: Class constructor cannot be invoked without 'new' - 无法直接在
class声明里写多个extends,这是语法错误
注意 Symbol.toStringTag 等不可枚举属性不会被 Object.assign 复制
如果某个 mixin 依赖 Symbol.toStringTag 或自定义 getter/setter,仅靠 Object.assign 会漏掉它们——因为该方法只复制可枚举自有属性。
立即学习“Java免费学习笔记(深入)”;
- 要用
Object.getOwnPropertyDescriptors+Object.defineProperties才能完整复制访问器和不可枚举属性 - 第三方库如
lodash.merge或deepMixIn类工具也常因忽略 symbol 属性导致行为异常 - 调试时发现方法存在但
console.dir(instance)看不到,很可能是用了不完整的拷贝方式
混入真正难的不是怎么写,而是厘清“谁负责清理副作用”“冲突方法由谁兜底”“是否要支持取消混入”。这些在大型项目里一旦没约定好,很快就会变成隐性耦合。
# javascript
# es6
# java
# 工具
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
🚀拖拽式CMS建站能否实现高效与个性化并存?
Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南
Laravel Eloquent模型如何创建_Laravel ORM基础之Model创建与使用教程
Laravel如何处理和验证JSON类型的数据库字段
最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?
Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复
JavaScript如何实现类型判断_typeof和instanceof有什么区别
在线制作视频网站免费,都有哪些好的动漫网站?
微信推文制作网站有哪些,怎么做微信推文,急?
利用python获取某年中每个月的第一天和最后一天
Laravel如何配置和使用缓存?(Redis代码示例)
Android 常见的图片加载框架详细介绍
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
Laravel如何实现密码重置功能_Laravel密码找回与重置流程
如何在阿里云部署织梦网站?
Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】
焦点电影公司作品,电影焦点结局是什么?
JavaScript Ajax实现异步通信
如何选择PHP开源工具快速搭建网站?
Laravel如何处理跨站请求伪造(CSRF)保护_Laravel表单安全机制与令牌校验
如何挑选高效建站主机与优质域名?
Laravel Blade模板引擎语法_Laravel Blade布局继承用法
Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程
Laravel如何优化应用性能?(缓存和优化命令)
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
百度浏览器如何管理插件 百度浏览器插件管理方法
Laravel如何实现数据导出到PDF_Laravel使用snappy生成网页快照PDF【方案】
Laravel用户认证怎么做_Laravel Breeze脚手架快速实现登录注册功能
Laravel中的withCount方法怎么高效统计关联模型数量
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
Laravel怎么实现微信登录_Laravel Socialite第三方登录集成
HTML透明颜色代码在Angular里怎么设置_Angular透明颜色使用指南【详解】
高配服务器限时抢购:企业级配置与回收服务一站式优惠方案
UC浏览器如何设置启动页 UC浏览器启动页设置方法
Laravel如何集成Inertia.js与Vue/React?(安装配置)
如何为不同团队 ID 动态生成多个非值班状态按钮
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
html如何与html链接_实现多个HTML页面互相链接【互相】
Laravel如何实现全文搜索功能?(Scout和Algolia示例)
Laravel如何生成和使用数据填充?(Seeder和Factory示例)
Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】
高防服务器租用首荐平台,企业级优惠套餐快速部署
Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全
Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
如何在IIS中新建站点并配置端口与物理路径?
Laravel的契約(Contracts)是什么_深入理解Laravel Contracts与依赖倒置
详解vue.js组件化开发实践
Linux网络带宽限制_tc配置实践解析【教程】


`); }
};