我有一个这样的课,我在自己的代码中创建了自己:
class StarryEyes { @Inject MyValidator validator; public StarryEyes(String name) { //.. } public doSomething() { // validator is NULL } }
我希望Guice注入一个验证器的实例,它有一个@Singleton注释.
我有一个在启动时加载的模块,它包含以下行:
bind(MyValidator.class);
但是,它似乎不起作用,因为“验证器”始终为空.我尝试了许多变体,如:
bind(MyValidator.class)toInstance(new MyValidator());
或其他类似的东西.这不是Guice应该如何工作吗?
解决方法
通常,Guice需要创建对象来注入它们.如果你刚刚打电话给新的StarryEyes(名字),Guice就不会去看那个对象,所以不能注入它.您可以做的一件事是在创建它之后,在对象上调用inject.injectMembers(obj).不过,我不建议您避免在代码中引用注入器.
你真正想要的是Assisted Inject.通过Assisted Inject,你可以为你的类声明一个这样的构造函数:
@Inject public StarryEyes(MyValidator validator,@Assisted String name)
这意味着验证器是Guice应该注入的参数,而name必须“辅助”(即在创建实例时提供).
然后,您将创建一个如下所示的界面:
public interface StarryEyesFactory { StarryEyes create(String name); }
通过Assisted Inject,Guice可以为您实施该工厂.你这样绑定:
bind(StarryEyesFactory.class).toProvider( FactoryProvider.newFactory(StarryEyesFactory.class,StarryEyes.class));
然后,您可以在任何要创建其实例的地方注入StarryEyesFactory.您以前会打电话给新的StarryEyes(名称),现在可以调用starryEyesFactory.create(name).当在工厂中调用create(name)时,它将使用名称并将其传递给构造函数,并提供绑定的验证器本身.
从Guice 3开始,您可以使用FactoryModuleBuilder:
install(new FactoryModuleBuilder().build(StarryEyesFactory.class));