我正在尝试使用InversifyJS在我的打字稿应用程序中进行依赖项注入。当我按照文档中给出的简单示例进行操作时,一切都将正常运行。但是,依赖关系不会注入到依赖关系链中较低的类中。希望这个例子可以使我想说的更加清楚。我尽可能地删除了样板:
EmployeeFactory.ts
@injectable()
class EmployeeFactory {
create(employeeName: string): Employee {
/*
* I think my error comes from the use of
* the new keyword right here. Not sure though.
*/
return new Employee(employeeName);
}
}
Company.ts
@injectable()
class Company {
@inject('EmployeeFactory') private employeeFactory: EmployeeFactory;
public getEmployees() {
return [
this.employeeFactory.create('Alice'),
this.employeeFactory.create('Bob')
]
}
}
Employee.ts
@injectable()
class Employee {
@inject('EmployeeFactory') private employeeFactory: EmployeeFactory;
constructor(readonly name: string) {
}
public getBoss(): Employee {
//TypeError: Cannot read property 'create' of undefined
return this.employeeFactory.create("Boss man")
}
}
然后这是我运行以获取错误的代码
import { Container } from "inversify";
let myContainer = new Container();
myContainer.bind<EmployeeFactory>("EmployeeFactory").to(EmployeeFactory);
myContainer.bind<Company>('Company').to(Company);
myContainer.bind<Employee>('Employee').to(Employee);
let oscorp: Company = myContainer.get<Company>('Company');
let employees: Employee[] = oscorp.getEmployees();
console.log('employees', employees);
// PRINTS: employees [ Employee { name: 'Alice' }, Employee { name: 'Bob' } ]
let boss = employees[0].getBoss();
console.log('boss', boss);
所以我的问题是:如何在整个应用程序中都没有“ myContainer”的情况下将EmployeeFactory注入到Employee类中?我确定我缺少有关InversifyJS的关键信息,因此可以提供任何帮助。
这里的问题是,类Employee
是仅使用运算符new
创建的,而不是使用inversifyJS创建的。因此,您仅通过JS创建新实例,而无需进行任何依赖项注入。
[如果您只想使代码正常工作,则只需重写代码即可
EmployeeFactory.ts
@injectable()
class EmployeeFactory {
create = (employeeName: string): Employee => {
return new Employee(this, employeeName);
}
}
Employee.ts
@injectable()
class Employee {
constructor(
private _employeeFactory: EmployeeFactory,
readonly name: string,
) {}
public getBoss(): Employee {
return this._employeeFactory.create("Boss man")
}
}
但是,如果要使其逆向化,则需要使用Factory Injection:https://github.com/inversify/InversifyJS/blob/master/wiki/factory_injection.md
然后您需要将绑定中的代码更改为
myContainer.bind<Company>('Company').to(Company);
myContainer.bind<Employee>('Employee').toConstructor(Employee);
container.bind<interfaces.Factory<Employee>>('Factory<Employee>').toFactory<Employee>((context: interfaces.Context) => {
return (name: string) => {
const EmployeeConstructor = context.container.get<Employee>('Employee');
return new EmployeeConstructor(name);
};
});
Company.ts
@injectable()
class Company {
@inject('Factory<Employee>') private employeeFactory: (name: string) => Employee;
public getEmployees() {
return [
this.employeeFactory('Alice'),
this.employeeFactory('Bob')
]
}
}