3

我有一个使用 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>

我的应用程序初始化:

@Override
protected void init()
{
    super.init();
    getResourceSettings().setResourcePollFrequency(null);
    getMarkupSettings().setStripWicketTags(true);
    getDebugSettings().setDevelopmentUtilitiesEnabled(true);
    GuiceComponentInjector injector = new GuiceComponentInjector(this, new WebModule(), new GuiceModule());;
}

自定义Servlet配置:

public class CustomServletConfig  extends GuiceServletContextListener {

    @Override
    protected Injector getInjector() {
        return Guice.createInjector(new GuiceModule(), new 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 启动我的应用程序),还是我的设计有问题?

4

1 回答 1

3

这是很常见的故障。

ServletScopes或中的所有实体RequestScopes都应作为Providers.

所以你的代码应该是:

@Inject
Provider<IUserService> userServiceProvider

public IUserService getUserService() {
  userServiceProvider.get(); 
}

为什么这样?!只要您在 Stage.DEVELOPMENT 中使用它并且不急于创建父类,一切都很好。如果您将父类绑定为asEagerSingleton或切换到 Stage.PRODUCTION,您的类将在启动时开始急切地创建。否则,它们仅在被访问时才以惰性方式创建(很可能在第一次请求期间)。

你的问题就出现了。您WebApplication在启动时急切地初始化。然后 guice 尝试注入所有子依赖项,并IUserServiceSessionScope. 问题是您当前不在 GuiceFilter 内并且没有请求,因此 guice 无法确定当前会话或创建新会话。因此无法达到这些范围。您目前在您的应用程序中,ContextListener并且您的应用程序正在热切地实例化。Singleton如果您只使用而不是asEagerSingleton因为延迟加载,一切都会好起来的。

无论如何,将 Session 和 Request 范围的对象作为 Providers 传递是最佳实践。你可以在这里了解更多关于ProvidersScopes 的信息(还有一个很好的表格,其中包含了急切与延迟加载的比较)

于 2013-10-30T10:57:04.790 回答