Bean是什么
我们知道Bean
是Spring
最基础的核心构件,大多数逻辑代码都通过Bean
进行管理。NestJS
基于TypeScript
和依赖注入
也实现了类似于Spring Bean
的机制:服务提供者(Provider)
CabloyJS则是在原生JS(Vanilla JS)
上实现了更轻量、更灵活的Bean容器
理念
CabloyJS在设计Bean容器机制时,遵循了以下3个理念:
1. 几乎所有事物都是Bean
我们绝大多数逻辑代码都通过Bean组件进行管理,比如:Controller、Service、Model、Middleware、Event、Queue、Broadcast、Schedule、Startup、Flow、Flow Task,等等
2. Bean支持AOP
所有Bean组件都可以通过AOP组件进行逻辑扩展
3. AOP也是一种Bean
AOP组件既然也是Bean,那么也可以通过其他AOP组件进行逻辑扩展
定义Bean
CabloyJS约定了两种定义Bean的模式:app和ctx。由于Bean被容器托管,可以很方便的跨模块调用。因此,为了清晰的辨识Bean被应用的场景,一般约定:如果Bean只被本模块内部调用,那么就使用app模式;如果大概率会被其他模块调用,那么就使用ctx模式
1. app模式
比如:Controller、Service都采用app模式
src/module/test-party/backend/src/bean/test.app.js
module.exports = app => {
class appBean extends app.meta.BeanBase {
actionSync({ a, b }) {
return a + b;
}
async actionAsync({ a, b }) {
return Promise.resolve(a + b);
}
}
return appBean;
};
2. ctx模式
比如:ctx.bean.atom
、ctx.bean.user
、ctx.bean.role
都采用ctx模式
src/module/test-party/backend/src/bean/test.ctx.js
module.exports = ctx => {
class ctxBean {
constructor(moduleName) {
this._name = moduleName || ctx.module.info.relativeName;
}
get name() {
return this._name;
}
set name(value) {
this._name = value;
}
actionSync({ a, b }) {
return a + b;
}
async actionAsync({ a, b }) {
return Promise.resolve(a + b);
}
}
return ctxBean;
};
注册Bean
对于大多数组件,EggJS采用约定优先
的策略,会在指定的位置查找资源,并自动加载。而CabloyJS采用显式注册
,从而Webpack可以收集所有后端源码,实现模块编译的特性
src/module/test-party/backend/src/beans.js
const testApp = require('./bean/test.app.js');
const testCtx = require('./bean/test.ctx.js');
module.exports = app => {
const beans = {
// test
'test.app': {
mode: 'app',
bean: testApp,
},
testctx: {
mode: 'ctx',
bean: testCtx,
global: true,
},
};
return beans;
};
名称 | 说明 | mode | 模式:app/ctx | bean | bean组件 | global | 是否是全局组件 |
---|
使用Bean
1. beanFullName
每一个注册的Bean组件都被分配了全称,具体规则如下
注册名称 | 场景 | 所属模块 | global | beanFullName | test.app | test | test-party | false | test-party.test.app | testctx | test-party | true | testctx |
---|
2. 基本调用
可以直接通过this.ctx.bean
取得Bean容器,然后通过beanFullName
获取Bean实例
src/module/test-party/backend/src/controller/test/feat/bean.js
// global: false
this.ctx.bean['test-party.test.app'].actionSync({ a, b });
await this.ctx.bean['test-party.test.app'].actionAsync({ a, b });
// global: true
this.ctx.bean.testctx.actionSync({ a, b });
await this.ctx.bean.testctx.actionAsync({ a, b });
3. 新建Bean实例
通过this.ctx.bean
获取Bean实例,那么这个实例对当前ctx
而言是单例的。如果需要新建Bean实例,可以按如下方式进行:
ctx.bean._newBean(beanFullName, ...args)
比如我们要新建一个Flow实例:
src/module-system/a-flow/backend/src/bean/bean.flow.js
_createFlowInstance({ flowDef }) {
const flowInstance = ctx.bean._newBean(`${moduleInfo.relativeName}.local.flow.flow`, {
flowDef,
});
return flowInstance;
}
4. 跨模块调用本地Bean
本地Bean也可以被跨模块调用
await ctx.executeBean({ locale, subdomain, beanModule, beanFullName, context, fn, transaction })
名称 | 可选 | 说明 | locale | 可选 | 默认等于ctx.locale | subdomain | 可选 | 默认等于ctx.subdomain | beanModule | 必需 | 本地Bean所属模块名称 | beanFullName | 必需 | 本地Bean的全称 | context | 可选 | 调用本地Bean时传入的参数 | fn | 必需 | 调用本地Bean的方法名 | transaction | 可选 | 是否要启用数据库事务 |
---|
比如我们要调用模块a-file
的本地Bean: service.file
,直接上传用户的avatar,并返回downloadUrl
src/module-system/a-base-sync/backend/src/bean/bean.user.js
// upload
const res2 = await ctx.executeBean({
beanModule: 'a-file',
beanFullName: 'a-file.service.file',
context: { fileContent: res.data, meta, user: null },
fn: '_upload',
});
// hold
profile._avatar = res2.downloadUrl;
5. app.bean
ctx.bean
是每个请求初始化一个容器,而app.bean
则可以实现整个应用使用一个容器,从而实现Bean组件的应用级别的单例模式
src/module/test-party/backend/src/controller/test/feat/bean.js
app.bean['test-party.test.app'].actionSync({ a, b });
await app.bean['test-party.test.app'].actionAsync({ a, b });
AOP编程
限于篇幅,关于AOP编程
请参见:cabloy-aop
相关链接
- [官网: cabloy.com/](https://c…
- [GitHub: github.com/zhennann/ca…
常见问题FAQ
- 免费下载或者VIP会员专享资源能否直接商用?
- 本站所有资源版权均属于原作者所有,这里所提供资源均只能用于参考学习用,请勿直接商用。若由于商用引起版权纠纷,一切责任均由使用者承担。更多说明请参考 VIP介绍。
- 提示下载完但解压或打开不了?
- 找不到素材资源介绍文章里的示例图片?
- 模板不会安装或需要功能定制以及二次开发?
发表评论
还没有评论,快来抢沙发吧!