如何在不影响原始bean的"客户端"的情况下声明另一个Jackson ObjectMap?

Tim*_*imi 6 java spring dependency-injection jackson spring-boot

我有一个spring-boot应用程序,它公开了一个json REST API.为了将对象映射到json,它使用spring-boot配置的内置jackson ObjectMapper.

现在我需要从yaml文件中读取一些数据,我发现一个简单的方法是使用Jackson - 为此我需要声明一个不同的ObjectMapper来将yaml转换为对象.我声明这个新的mapper bean有一个特定的名称,可以在我的服务中注入它来处理从yaml文件中读取:

@Bean(YAML_OBJECT_MAPPER_BEAN_ID)
public ObjectMapper yamlObjectMapper() {
    return new ObjectMapper(new YAMLFactory());
}
Run Code Online (Sandbox Code Playgroud)

但我需要一种方法来告诉原始json ObjectMapper的所有其他"客户端"继续使用该bean.所以基本上我需要在原始bean上使用@Primary注释.有没有办法实现这一点,而无需在我自己的配置中重新声明原始的ObjectMapper(我必须挖掘spring-boot代码来查找和复制其配置)?

我找到的一个解决方案是为ObjectMapper声明一个FactoryBean并使它返回已经声明的bean,如本答案所示.我通过调试发现我的原始bean被称为"_halObjectMapper",所以我的factoryBean将搜索这个bean并返回它:

public class ObjectMapperFactory implements FactoryBean<ObjectMapper> {

    ListableBeanFactory beanFactory;

    public ObjectMapper getObject() {
        return beanFactory.getBean("_halObjectMapper", ObjectMapper.class);
    }
    ...
}
Run Code Online (Sandbox Code Playgroud)

然后在我的Configuration类中,我将其声明为@Primary bean,以确保它是自动装配的首选:

@Primary
@Bean
public ObjectMapperFactory objectMapperFactory(ListableBeanFactory beanFactory) {
    return new ObjectMapperFactory(beanFactory);
}
Run Code Online (Sandbox Code Playgroud)

尽管如此,我对这个解决方案并不十分满意,因为它依赖于不受我控制的bean的名称,而且它看起来像是一个黑客.有更清洁的解决方案吗?

谢谢!

Dar*_*hta 5

您可以定义两个ObjectMapperbean 并将其中一个声明为主要 bean,例如:

@Bean("Your_id")
public ObjectMapper yamlObjectMapper() {
    return new ObjectMapper(new YAMLFactory());
}

@Bean
@Primary
public ObjectMapper objectMapper() {
    return new ObjectMapper();
}
Run Code Online (Sandbox Code Playgroud)

完成后,您可以使用带有@Qualifier注释的 objectmapper bean ,例如:

@Autowired
@Qualifier("Your_id")
private ObjectMapper yamlMapper;
Run Code Online (Sandbox Code Playgroud)

更新

您可以ObjectMapper在运行时动态地将您的添加到 Spring 的 bean 工厂,例如:

@Configuration
public class ObjectMapperConfig {

    @Autowired
    private ConfigurableApplicationContext  context;

    @PostConstruct
    private void init(){
        BeanDefinitionBuilder builder = BeanDefinitionBuilder.rootBeanDefinition(ObjectMapper.class);
        builder.addConstructorArgValue(new JsonFactory());
        DefaultListableBeanFactory factory = (DefaultListableBeanFactory) context.getBeanFactory();
        factory.registerBeanDefinition("yamlMapper", builder.getBeanDefinition());
        Map<String, ObjectMapper> beans = context.getBeansOfType(ObjectMapper.class);
        beans.entrySet().forEach(System.out::println);
    }
}
Run Code Online (Sandbox Code Playgroud)

上面的代码在context不改变现有bean的情况下添加了一个新bean (sysoutinit方法末尾打印两个bean )。然后,您可以使用“yamlMapper”作为限定符在任何地方自动装配它。

更新 2(来自问题作者):

“更新”中建议的解决方案有效,这是一个简化版本:

@Autowired
private DefaultListableBeanFactory beanFactory;

@PostConstruct
private void init(){
    BeanDefinitionBuilder builder = BeanDefinitionBuilder.rootBeanDefinition(YAMLMapper.class);
    beanFactory.registerBeanDefinition("yamlMapper", builder.getBeanDefinition());
}
Run Code Online (Sandbox Code Playgroud)