Typescript 依赖注入:注入NPM模块的推荐模式
我想使用Inversify删除NPM模块上的硬依赖项,并将它们作为构造函数参数注入。在我尝试之前,这似乎要简单得多 事实证明,大多数确定类型的模块不需要麻烦地导出接口,当它们导出接口时,很少包含代表整个模块的接口。此外,在导出类时,我仍然必须手动为该类定义构造函数接口 这意味着我几乎每个模块都要做类似的事情:Typescript 依赖注入:注入NPM模块的推荐模式,typescript,inversifyjs,Typescript,Inversifyjs,我想使用Inversify删除NPM模块上的硬依赖项,并将它们作为构造函数参数注入。在我尝试之前,这似乎要简单得多 事实证明,大多数确定类型的模块不需要麻烦地导出接口,当它们导出接口时,很少包含代表整个模块的接口。此外,在导出类时,我仍然必须手动为该类定义构造函数接口 这意味着我几乎每个模块都要做类似的事情: import * as ConcreteModule from 'module' import { ContainerModule } from 'inversify' export i
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)
})
但是,只有当模块导出一个类(不是工厂)并且仍然不能真正帮助我构造构造函数时,这才有效。下面的示例演示如何将npm模块(
lodash
&sequelize
)注入类SomeClass
)
目录结构如下所示:
src/
├── entities
│ └── some_class.ts
├── index.ts
└── ioc
├── interfaces.ts
├── ioc.ts
└── types.
/src/ioc/types.ts
/src/ioc/interfaces.ts
/src/ioc/ioc.ts
/src/index.ts
/tsconfig.json
这个示例现在可以在中找到。不幸的是,DT中的键入应该首先导出这些接口。没有什么能阻止他们这样做,这只是不正确的做法。我们需要一次换一个DT打字机来解决这个问题。谢谢!这正是我想要的。是时候更彻底地阅读TypeScript规范了。。。我不知道存在类型查询,而且它们似乎根本没有记录在typescriptlang.org上的手册中。@Remo根据您的回答,在哪里配置sequelize,即
new sequelize('database'、'username'、'password'、…
insideSomeClass
?
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
}
}