我们正在使用GIN-in-out GWT项目进行评估,并且通过构造函数参数的典型注入取得了良好的结果。我们遇到的困难是场级注入。字段最终总是为null。有人有一个很好的例子来说明如何使用GIN正确地实现文件级注入吗?
更新:
下面是一些类似于我们的示例代码:
public class MVP implements EntryPoint {
public static final HandlerManager EVENT_BUS = new HandlerManager(null);
private final MVPInjector _injector = GWT.create(MVPInjector.class);
public void onModuleLoad() {
// set up layout for module
RootLayoutPanel.get().add(mainPanel);
// initialize presenters
ListPresenter listPresenter = _injector.listPresenter();
DetailPresenter detailPresenter = _injector.detailPresenter();
listPresenter.go(listContainer);
detailPresenter.go(detailContainer);
// simulate data coming in from RPC call
EVENT_BUS.fireEvent(new DataReadyEvent(getData()));
}
}
public class ListPresenter {
private final HandlerManager _eventBus;
private final Map<String, Fruit> _myRecords = new HashMap<String, Fruit>();
private final Display _view;
@Inject
public ListPresenter(Display argView, HandlerManager argEventBus) {
_eventBus = argEventBus;
_view = argView;
}
public void go(HasWidgets argContainer) {
argContainer.clear();
argContainer.add(_view.asWidget());
}
public interface Display {
public Widget asWidget();
public void clear();
public SingleSelectionModel<ViewProxy> getSelectionModel();
public void setData(List<ViewProxy> argData);
}
}
public class DetailPresenter {
private final HandlerManager _eventBus;
private final Display _view;
private Fruit _myRecord;
@Inject
private ImagePresenterFactory _imagePresenterFactory;
@Inject
private TestPresenter _testPresenter;
@Inject
public DetailPresenter(Display argView, HandlerManager argEventBus) {
_view = argView;
_eventBus = argEventBus;
}
public void go(HasWidgets argContainer) {
argContainer.clear();
argContainer.add(_view.asWidget());
if (_testPresenter != null) {
_testPresenter.go();
}
}
public interface Display {
public Widget asWidget();
public HasText getDescriptionControl();
public HasClickHandlers getImageControl();
public HasText getNameControl();
public HasClickHandlers getSaveControl();
public void setEnabledControls(boolean argEnabled);
}
}
public class TestPresenter {
@Inject
HandlerManager _eventBus;
public TestPresenter() {}
public void go() {
if (_eventBus != null) {
_eventBus.toString();
}
else {
// event bus was not injected
}
}
}
@GinModules(MVPModule.class)
public interface MVPInjector extends Ginjector {
DetailPresenter detailPresenter();
ListPresenter listPresenter();
}
public class MVPModule extends AbstractGinModule {
@Provides
@Singleton
public HandlerManager getEventBus() {
return MVP.EVENT_BUS;
}
@Provides
public TestPresenter getTestPresenter() {
return new TestPresenter();
}
@Override
protected void configure() {
bind(ListPresenter.Display.class).to(ListView.class);
bind(DetailPresenter.Display.class).to(DetailView.class);
bind(ImagePresenter.Display.class).to(ImagePopup.class);
install(new GinFactoryModuleBuilder().build(ImagePresenterFactory.class));
}
public interface ImagePresenterFactory {
public ImagePresenter createImagePresenter(ImageResource argImage);
}
}
在上面的代码中,我删除了大部分不涉及GIN的代码。DetailPresenter所需的TestPresenter已成功注入,但TestPresenter所要求的HandlerManager始终为null。正如您所看到的,在构造函数中没有使用注入的HandlerManager。
更新,查看示例代码:
@Provides
public TestPresenter getTestPresenter() {
return new TestPresenter();
}
因为你是自己创造的,它假设你已经处理过任何注射。删除此方法,它将调用默认构造函数(如果需要,在那里注入),然后访问任何其他注入站点。
您可能遇到的另一个问题是:有几个HandlerManager impl,请确保您对HandlerManager的所有引用都使用相同的包。
原始答案:
当构造函数运行时,它们将为null,但这是有道理的——当注入器还没有机会分配所有字段时,它们怎么可能是任何其他值。考虑一下它是如何运行的(这里用合法的java表示,因为字段可能不是公共的):
InstanceToInject instance = new InstanceToInject(...);
instance.field = provideFieldValue();
当字段有资格被分配时,您的构造函数已经运行。
如果在运行另一个方法时该字段为null,请确保该方法不是由构造函数运行的,而是在注入完成其工作之后运行的。其他可能为null的情况是@Inject
注释的setter。
假设不是这些情况(最简单的检查方法是设置断点,并确保注入器不在调用堆栈中),请确保字段确实有@Inject
,并且它没有绑定到null实例。