- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我想使用 Inversify 移除对 NPM 模块的硬依赖,并将它们作为构造函数参数注入(inject)。在我试一试之前,这似乎要简单得多。
事实证明,大多数 DefinitelyTyped 模块都懒得导出接口(interface),即使它们导出接口(interface),也很少包含代表整个模块的接口(interface)。此外,当一个类被导出时,我仍然必须手动为该类定义一个构造函数接口(interface)。
这意味着我必须为几乎每个模块做这样的事情:
import * as ConcreteModule from 'module'
import { ContainerModule } from 'inversify'
export interface ModuleInstance {
// copy a ton of stuff from DefinitelyTyped repo,
// because they didn't export any interfaces
}
export interface ModuleConstructor {
new (...args: any[]): ModuleInstance
}
export const ModuleConstructorSymbol = Symbol('ModuleConstructor')
export const ModuleContainer = new ContainerModule((bind) => {
bind<ModuleConstructor>(ModuleConstructorSymbol).toConstantValue(ConcreteModule)
})
有什么方法可以简化其中的一些吗?注入(inject) NPM 模块的开销非常大,Inversify 文档中没有任何指导。管理您需要的所有不同导入/导出(接口(interface)、符号和容器)的名称是一件痛苦的事情,需要提出某种一致的命名方案。似乎没有 TypeScript 支持从模块自动创建接口(interface)的某种方式,就没有办法以理智的方式注入(inject) NPM 包。
我想我可以只对模块使用 jest 的自动模拟功能,但我真的不喜欢将我的代码设计成只能使用特定测试框架进行单元测试的方式。
如果我能做到这一点,这似乎至少更容易实现:
import * as ConcreteModule from 'module'
export interface TheModule extends ConcreteModule {}
但这仅在模块导出一个类(不是工厂)并且仍然不能真正帮助我使用构造函数时才有效。
最佳答案
以下示例演示了如何将 npm 模块(lodash
和 sequelize
)注入(inject)到类 SomeClass
中。
目录结构如下:
src/
├── entities
│ └── some_class.ts
├── index.ts
└── ioc
├── interfaces.ts
├── ioc.ts
└── types.
const TYPES = {
Sequelize: Symbol("Sequelize"),
Lodash: Symbol("Lodash"),
SomeClass: Symbol("SomeClass")
};
export { TYPES };
import * as sequelize from "sequelize";
import * as _ from "lodash";
export type Sequelize = typeof sequelize;
export type Lodash = typeof _;
export interface SomeClassInterface {
test(): void;
}
import { Container, ContainerModule } from "inversify";
import * as sequelize from "sequelize";
import * as _ from "lodash";
import { TYPES } from "./types";
import { Sequelize, Lodash } from "./interfaces";
import { SomeClass } from "../entities/some_class";
const thirdPartyDependencies = new ContainerModule((bind) => {
bind<Sequelize>(TYPES.Sequelize).toConstantValue(sequelize);
bind<Lodash>(TYPES.Lodash).toConstantValue(_);
// ..
});
const applicationDependencies = new ContainerModule((bind) => {
bind<SomeClass>(TYPES.SomeClass).to(SomeClass);
// ..
});
const container = new Container();
container.load(thirdPartyDependencies, applicationDependencies);
export { container };
import { Container, injectable, inject } from "inversify";
import { TYPES } from "../ioc/types";
import { Lodash, Sequelize, SomeClassInterface } from "../ioc/interfaces";
@injectable()
class SomeClass implements SomeClassInterface {
private _lodash: Lodash;
private _sequelize: Sequelize;
public constructor(
@inject(TYPES.Lodash) lodash,
@inject(TYPES.Sequelize) sequelize,
) {
this._sequelize = sequelize;
this._lodash = lodash;
}
public test() {
const sequelizeWasInjected = typeof this._sequelize.BIGINT === "function";
const lodashWasInjected = this._lodash.cloneDeep === "function";
console.log(sequelizeWasInjected); // true
console.log(lodashWasInjected); // true
}
}
export { SomeClass };
import "reflect-metadata";
import { container } from "./ioc/ioc";
import { SomeClassInterface } from "./ioc/interfaces";
import { TYPES } from "./ioc/types";
const someClassInstance = container.get<SomeClassInterface>(TYPES.SomeClass);
someClassInstance.test();
{
"name": "test",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"license": "ISC",
"dependencies": {
"inversify": "^4.1.0",
"lodash": "^4.17.4",
"reflect-metadata": "^0.1.10",
"sequelize": "^3.30.4"
},
"devDependencies": {
"@types/lodash": "^4.14.63",
"@types/sequelize": "^4.0.51"
}
}
{
"compilerOptions": {
"target": "es5",
"lib": ["es6", "dom"],
"types": ["reflect-metadata"],
"module": "commonjs",
"moduleResolution": "node",
"experimentalDecorators": true,
"emitDecoratorMetadata": true
}
}
此示例现已在 inversify docs 中可用.
关于typescript - 依赖注入(inject) : recommended pattern for injecting NPM modules,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43669697/
这个问题在这里已经有了答案: Difference between / and /* in servlet mapping url pattern (5 个回答) 4年前关闭。 web.xml 中的/
Scala 具有支持模式匹配中析取的语言功能(“模式替代”): x match { case _: String | _: Int => case _ => } 但是,如果审查满足 P
解释我的问题: 类别:玩具 特质 1:说话像男性 特质2:说话像女性 我能否在运行时更改 Toy 的行为(特征),以便有时同一个对象说话像男性,有时同一个对象说话像女性? 我想在运行时改变说话行为。
我已经能够找到很好的资源,这些资源告诉我 Java API 中的 MouseAdapter 没有使用适配器模式。问题是:MouseAdapter 是否实现了某种模式? 我知道它的作用:它为 Mouse
我有兴趣了解有关模式识别的更多信息。我知道这是一个广泛的领域,所以我将列出一些我想学习处理的特定类型的问题: 在看似随机的字节集中查找模式。 识别图像中的已知形状(例如圆形和正方形)。 注意给定位置流
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 8 年前。 Improve this ques
所以,问题很简单:在 awk 中,if (var ~/pattern/) 是否与 if (var ~ "pattern") 相同? 我已经对 csv 进行了一些基本测试,两者似乎都产生了相同的结果..
我的问题是 this 的 Scala (Java) 变体Python 上的查询。 特别是,我有一个字符串 val myStr = "Shall we meet at, let's say, 8:45
我最近一直在研究正则表达式并注意到了这一点。 Pattern pNoEmbed = Pattern.compile("[ a-z]+", Pattern.CASE_INSENSITIVE); Patt
在研究大型应用程序的 C++ 源代码时,我发现了这种模式(该示例的语法可能很粗略,但基本细节都在那里): class A : X friend B; B *parent; ...
有人可以举一个“中介者模式”在现实世界中有用的用例吗? 最佳答案 Mediator是一种添加第三方对象以控制一组(2 个或更多)对象之间交互的方法。 您能找到的最简单的示例是 Chat Room例如,
尝试编译以下代码片段时: type 'a frame = Empty | Frame of string * 'a * 'a frame let rec searchFrame f s = match
目标 我的目标是获得一个 servlet 过滤器来处理对主页的请求,然后再将它们转发到 index.jsp。 问题 我无法让过滤器接收来自“/”的请求。它的 URL 模式是 / 相反,对该模式的请求最
这个问题已经有答案了: Difference between / and /* in servlet mapping url pattern (5 个回答) 已关闭 6 年前。 我已经设置了一个具有此
第 6 章(代码重用模式)中有以下示例: // the parent constructor function Parent(name) { this.name = name || 'Adam
Pattern类中的pattern()方法和toString()方法有什么区别? 文档说: public String pattern() Returns the regular expression
我有脚本 here并且 ng-pattern 工作正常,因为 scope.subnet 仅在输入匹配模式后才显示在输出中。但是如果 ng-pattern 不匹配,ng-show 不会显示任何错误
我想知道为什么当提供相同的正则表达式和相同的字符串时,java regex pattern.matcher() 和 pattern.matches() 的结果会不同 String str = "hel
This SO answer引用“患有模式综合症的小男孩”。虽然我可以通过上下文推断出一些含义,但我并不完全理解。 “有模式综合症的小男孩”的良好定义是什么? 最佳答案 它只是意味着寻找将模式注入(i
我有以下微服务架构的用例。 我的问题是,在当前情况下,我有 3 个微服务和一个 APIGateway。 最后,网关必须在聚合(组合)来自 3 个服务的数据之前进行大量查询。因为这 3 个微服务只提供基
我是一名优秀的程序员,十分优秀!