我有一个使用Guice [v3]的Wicket [v6]应用程序 - 我现在已经使用依赖注入存储库操作,我想将它用于使用会话范围的服务(每个用户的会话一个)。我在这里阅读了官方文档,各种博客文章和问题,但我不确定我是否使用了正确的方法。
我有两个问题: 1.我使用正确的方法吗? 2.在依赖SessionScoped注入的类上运行TestNG测试需要什么特殊的东西吗?
我的设置: web.xml中:
<filter>
<filter-name>guiceFilter</filter-name>
<filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>guiceFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<listener>
<listener-class>com.xxx.CustomServletConfig</listener-class>
MyApplication init:
@Override
protected void init()
{
super.init();
getResourceSettings().setResourcePollFrequency(null);
getMarkupSettings().setStripWicketTags(true);
getDebugSettings().setDevelopmentUtilitiesEnabled(true);
GuiceComponentInjector injector = new GuiceComponentInjector(this, new WebModule(), new GuiceModule());;
}
CustomServletConfig:
public class CustomServletConfig extends GuiceServletContextListener {
@Override
protected Injector getInjector() {
return Guice.createInjector(new GuiceModule(), new WebModule());
}
WebModule:
public static class WebModule extends ServletModule {
@Override
protected void configureServlets() {
bind(WebApplication.class).toProvider(WicketGuiceAppProvider.class).asEagerSingleton();
bind(IUserService.class).to(UserService.class).in(ServletScopes.SESSION);
Map<String, String> params = new HashMap<String, String>();
params.put(WicketFilter.FILTER_MAPPING_PARAM, "/*");
filter("/*").through(WicketGuiceFilter.class, params);
}
}
在示例页面中,我有:
@Inject
IUserService userService
...
userService.doSomething
在单元测试期间的userService.doSomething我收到Guice OutOfScopeException,指向我在ServletModule中的绑定: 自定义提供程序中的错误,com.google.inject.OutOfScopeException?:无法访问作用域对象。我们目前不在HTTP Servlet请求中,或者您可能忘记应用com.google.inject.servlet.GuiceFilter?作为此请求的servlet过滤器。
我的配置是否正常,我需要以不同方式运行单元测试(我只是使用WicketTester启动我的应用程序),还是我的设计有问题?
答案 0 :(得分:3)
这是非常常见的错误。
ServletScopes
或RequestScopes
中的所有实体都应以Providers
传递。
所以你的代码应该是:
@Inject
Provider<IUserService> userServiceProvider
public IUserService getUserService() {
userServiceProvider.get();
}
为什么这样?!只要你在Stage.DEVELOPMENT中使用它就一切都很好,并且不会急切地创建父类。如果将父类绑定为asEagerSingleton
或切换到Stage.PRODUCTION,则会在启动时急切地创建类。否则,只有在访问它们时才会以懒惰方式创建它们(很可能在第一次请求期间)。
你的问题出现了。您的WebApplication
在启动时急切地初始化。然后guice尝试注入所有子依赖项,并在IUserService
中找到SessionScope
字段注入。问题是你目前不在GuiceFilter里面并且没有请求,因此guice无法确定当前会话或创建新会话。所以这些范围无法达成。您当前在ContextListener
,并且您的应用程序急切地被实例化。如果由于延迟加载而只使用Singleton
而不是asEagerSingleton
,那么一切都会很好。
无论如何,将Session和Request范围对象作为提供者传递是最佳实践。您可以详细了解Providers here和Scopes here (there is also nice table with eager vs. lazy loading comparsion)