我似乎无法正确注册我的Jackson ObjectMapper模块。

我正在使用Guice + Jersey + Jackson(FasterXML)堆栈。

我遵循了如何根据此处的各种问题来自定义ObjectMapper的方法。特别是,我声明了一个ContextResolver,分别标记为@ javax.ws.rs.ext.Provider和@ javax.inject.Singleton。

我有一个类似的GuiceServletContextListener:

@Override
protected Injector getInjector() {

     Injector injector = Guice.createInjector(new DBModule(dataSource),
            new ServletModule()
            {
                @Override
                protected void configureServlets() {


                    // Mapper
                    bind(JacksonOMP.class).asEagerSingleton();

                    // ...

                    Map<String, String> initParams = new HashMap<String, String>();
                    initParams.put("com.sun.jersey.config.feature.Trace",
                            "true");
                    initParams.put("com.sun.jersey.api.json.POJOMappingFeature", "true");

                    serve("/services/*").with(
                            GuiceContainer.class,
                            initParams);
                }
            });

    return injector;
}

映射器定义
import javax.inject.Singleton;
import javax.ws.rs.Produces;
import javax.ws.rs.ext.ContextResolver;
import javax.ws.rs.ext.Provider;

@Provider
@Singleton
@Produces
public class JacksonOMP implements ContextResolver<ObjectMapper> {

  @Override
  public ObjectMapper getContext(Class<?> aClass) {
    final ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new Hibernate4Module());
    return mapper;
  }
}

但是,仅凭此配置,就不会调用getContext(),因此永远不会注册映射器。我陷入了典型的困境-注释-奥秘,在这里我实际上应该做的事几乎无法追查。 Spring用户只是报告说注册了该组件,而容器只是将其拾取。

This的答案讨论了重写我自己的javax.ws.rs.core.Application实现。但是,在GuiceContainer的jersey-guice实现中,这看起来像是硬连接为DefaultResourceConfig():
@Override
protected ResourceConfig getDefaultResourceConfig(Map<String, Object> props,
        WebConfig webConfig) throws ServletException {
    return new DefaultResourceConfig();
}

我应该在这里子类GuiceContainer吗?还是我缺少其他魔术注释?

这似乎是一件很普通的事情-我对使用这种guice组合被证明多么困难感到惊讶。

最佳答案



您确实应该阅读excellent Guice documentation。 Guice非常易于使用,它具有很少的基本概念。您的问题在于您混合了Jersey JAX-RS依赖项注入(inject)和Guice依赖项注入(inject)。如果您使用的是GuiceContainer,则声明您将对所有DI使用Guice,因此必须使用Guice(而不是JAX-RS)添加绑定(bind)。

例如,您不需要ContextResolver,应改用普通的Guice Provider:

import com.google.inject.Provider;

public class ObjectMapperProvider implements Provider<ObjectMapper> {
    @Override
    public ObjectMapper get() {
        final ObjectMapper mapper = new ObjectMapper();
        mapper.registerModule(new Hibernate4Module());
        return mapper;
    }
}

然后,您应该在模块中添加相应的绑定(bind):
bind(ObjectMapper.class).toProvider(ObjectMapperProvider.class).in(Singleton.class);

这将绑定(bind)ObjectMapper,但是不足以将Jersey与 jackson 一起使用。您将需要某种MessageBodyReader/MessageBodyWriter,例如JacksonJsonProvider。您将需要其他提供者:
public class JacksonJsonProviderProvider implements Provider<JacksonJsonProvider> {
    private final ObjectMapper mapper;

    @Inject
    JacksonJsonProviderProvider(ObjectMapper mapper) {
        this.mapper = mapper;
    }

    @Override
    public JacksonJsonProvider get() {
        return new JacksonJsonProvider(mapper);
    }
}

然后将其绑定(bind):
bind(JacksonJsonProvider.class).toProvider(JacksonJsonProviderProvider.class).in(Singleton.class);

这就是您需要做的所有事情-不需要子类化。

不过,还有一个空间可以优化代码大小。如果我是你,我将使用@Provides -methods:
@Provides @Singleton
ObjectMapper objectMapper() {
    final ObjectMapper mapper = new ObjectMapper();
    mapper.registerModule(new Hibernate4Module());
    return mapper;
}

@Provides @Singleton
JacksonJsonProvider jacksonJsonProvider(ObjectMapper mapper) {
    return new JacksonJsonProvider(mapper);
}

这些方法应添加到您的模块之一,例如匿名ServletModule。然后,您将不需要单独的提供程序类。
顺便说一句,您应该使用JerseyServletModule而不是普通的ServletModule,它为您提供了许多有用的绑定(bind)。

10-07 19:43
查看更多