Spring之坑:自定义Jackson的ObjectMapper,实现@ResponseBody的自定义渲染


XML

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean id="bean1" class="test.Bean1" />
    <bean id="bean2" class="test.Bean2" />

</beans>

通常,我们可以采用下面的Java Config方式代替上面的Xml,实现 fine-grained(细粒度) 配置。

Java Config

package test;

@Configuration
public class WebConfig {
  @Bean
  public Bean1 bean1(){
       //……
  }
  @Bean
  public Bean2 bean2(){
       //……
  }
}

XXXConfigurer

但是,有时候我们希望: 一组功能相关的Bean之间能够建立更直接明确的关系

  • 那么我们可以选择实现 XXXConfigurer 这类 回调接口,然后使用 @Configuration注解该实现类,并Override它们的抽象方法。例如:
功能回调接口
缓存org.springframework.cache.annotation.CachingConfigurer
定时任务org.springframework.scheduling.annotation.SchedulingConfigurer
异步(并发)任务org.springframework.scheduling.annotation.AsyncConfigurer
Spring MVC高级配置org.springframework.web.servlet.config.annotation.WebMvcConfigurer

注意:

  • XXXConfigurer 接口的实现类无疑需要复写其全部抽象方法(Java8之前,Spring旧版本),但是如果不希望覆盖默认增加额外配置:
    • 方法有返回值,则 return null
    • 方法无返回值,则不写任何实现代码
  • 当然你也可以直接继承其抽象适配器 XXXConfigurerAdapter ,根据配置需要复写方法。
  • 当然Java8以后,接口方法有了默认实现default,在新版的Spring中,你可以直接实现 XXXConfigurer 接口,并根据配置复写方法。

示例:WebMvcConfigurer

我们以 WebMvcConfigurer 为例:自定义Jackson的ObjectMapper,实现@ResponseBody的自定义渲染?

  • 解决方法是配置Spring MVC的 HttpMessageConverter 消息转换器
先来看xml方式
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"          
	xmlns:mvc="http://www.springframework.org/schema/mvc"
	xmlns:p="http://www.springframework.org/schema/p"
	xsi:schemaLocation="http://www.springframework.org/schema/beans 
		http://www.springframework.org/schema/beans/spring-beans-4.3.xsd 
		http://www.springframework.org/schema/mvc 
		http://www.springframework.org/schema/mvc/spring-mvc-4.3.xsd">
		
		<mvc:annotation-driven >
			<mvc:message-converters>
        		<bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
            		<property name="objectMapper" ref="objectMapper"/>
        		</bean>
       	    </mvc:message-converters>
		</mvc:annotation-driven>

     	<bean id="objectMapper" class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean"
           p:serializationInclusion="NON_NULL"/>
WebMvcConfigurer的抽象适配器

我们还可以通过继承 WebMvcConfigurer 的抽象适配器 org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter
来实现Spring MVC在Java Config 形式下的高级配置:

@Configuration
@EnableWebMvc
public class WebConfiguration extends WebMvcConfigurerAdapter {

	    @Override
	    public void configureMessageConverters(List<HttpMessageConverter<?>> converters) {
	        ObjectMapper mapper = new ObjectMapper();
	        mapper.setDefaultPropertyInclusion(Include.NON_NULL);
	        converters.add(new MappingJackson2HttpMessageConverter(mapper));
	    }
}

@EnableWebMvc
  • 到这里,我们已经实现了Spring MVC在Java Config 形式下的高级配置,但是需要注意,这里我们使用了 @EnableWebMvc
@EnableWebMvc源码
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(DelegatingWebMvcConfiguration.class)
public @interface EnableWebMvc {
}
  • 第4行,导入了配置类org.springframework.web.servlet.config.annotation.DelegatingWebMvcConfiguration
DelegatingWebMvcConfiguration源码
@Configuration
public class DelegatingWebMvcConfiguration extends WebMvcConfigurationSupport {
    
    @Autowired(required = false)
 	public void setConfigurers(List<WebMvcConfigurer> configurers) {
		if (!CollectionUtils.isEmpty(configurers)) {
			this.configurers.addWebMvcConfigurers(configurers);
		}
	}
}
  • 可以看到,DelegatingWebMvcConfiguration的作用就是注入WebMvcConfigurer的实现类, 而 DelegatingWebMvcConfiguration 又需要@EnableWebMvc来导入。所以如果你没有使用Spring Boot,而是传统的
    Spring项目,又想要使用WebMvcConfigurer来实现细粒度配置,你需要@EnableWebMvc

  • 官方文档上给出了另一种解决方式:

    If WebMvcConfigurer does not expose some advanced setting that needs to be configured,
    consider removing the @EnableWebMvc annotation and extending directly from
    WebMvcConfigurationSupport or DelegatingWebMvcConfiguration,

    如果WebMvcConfigurer没能解决你的需求,那么你可以考虑移除 @EnableWebMvc并且直接继承WebMvcConfigurationSupport or DelegatingWebMvcConfiguration

注意:

Spring boot已经通过Spring MVC的自动配置类WebMvcAutoConfiguration导入了DelegatingWebMvcConfiguration,所以不再需要@EnableWebMvc

  • If you want to take complete control of Spring MVC, you can add your own @Configuration annotated with @EnableWebMvc.

相对的,如果你使用了@EnableWebMvc,那么Spring MVC的自动配置将被忽略。


@EnableXXX与<xxx:annotation-driven>的冲突

对于传统Spring项目,Java Config的使用势必导致Xml 与Java Config 同时存在于项目中。这时候就需要我们去考虑Xml与annotation-based之间是否会产生冲突。比如下面这些基本功能相同标签注解

  • 这里强调基本功能,因为Java Config的配置更细粒度,自然容易获得其他扩展导致某些功能不一致。例如:@EnableWebMvc导入了DelegatingWebMvcConfiguration,从而实现了WebMvcConfigurer接口的注入。
<cache:annotation-driven /> <!--@EnableCaching--> 
<task:annotation-driven scheduler=""/> <!--@EnableScheduling--> 
<task:annotation-driven executor=""/> <!--@EnableAsync--> 
<mvc:annotation-driven /> <!--@EnableWebMvc--> 
<!-- …… 等等,诸如此类 -->

这里还是以<mvc:annotation-driven /> @EnableWebMvc为例,如果二者同时存在:

  • 传统Spring项目web.xml加载dispatcher-servlet.xml,毫无疑问配置在xml中的<mvc:annotation-driven />将会优先生效。
  • 所以,此时 @EnableWebMvc不生效,WebMvcConfigurer的实现类将不会被注入,Java Config的配置方式不会生效。
  • 所以,这种情况需要二选一。

至于,其他**XXXConfigurer** 在传统Spring项目中的配置方式,与示例同理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值