Spring Cloud+Spring Boot使用Feign日期参数转化异常源码追踪及解析

Spring Cloud+Spring Boot使用Feign日期参数转化异常源码追踪及解析

  • 问题来源
  • 源码追踪
  • 解决方式

问题来源

作者在使用Spring Cloud + Feign,在进行服务间接口调用时,发现在有RequestParam注解的时间参数时,Feign解析出来的HTTP请求参数会出现类似“2019-04-10 下午34:39”的错误样式,导致Feign Client接收RequestParam参数的时候无法解析时间报错。

源码追踪

Feign在进行参数解析的时候会通过SynchronousMethodHandler类的invoke方法来完成(具体寻找方法还得看源码慢慢找,我也是通过Google和断点慢慢找到的)
Spring Cloud+Spring Boot使用Feign日期参数转化异常源码追踪及解析_第1张图片
继续跟踪代码,发现其本质还是调用converter来转换参数。并且通过Debug可以发现其内置了一些converters,且cache里应该是根据参数类型挑选出来的对应converter。
Spring Cloud+Spring Boot使用Feign日期参数转化异常源码追踪及解析_第2张图片
到下面这步代码基本上可以确定是因为converter缺少的原因而导致LocalDateTime无法转换成对应格式的参数传到FeignClient了。
Spring Cloud+Spring Boot使用Feign日期参数转化异常源码追踪及解析_第3张图片

解决方式

最快的解决方式即是直接配置相关的Converter的Bean到Spring容器。在这里一个是目标类型,一个是源类型,作者这里只是在解决自己问题的时候写的Java8时间的转换器,大家可以根据自己的问题更改规则。

@Configuration
public class JavaTimeConverterConfiguration {
	@Bean
	public FeignFormatterRegistrar localDataTimeFormatRegister() {
		return registry -> registry.addConverter(new LocalDateTime2StringConverter());
	}

	@Bean
	public FeignFormatterRegistrar localDataFormatRegister() {
		return registry -> registry.addConverter(new LocalDate2StringConverter());
	}


	private class LocalDateTime2StringConverter implements Converter<LocalDateTime, String> {
		@Override
		public String convert(@NonNull LocalDateTime source) {
			return source.format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
		}
	}

	private class LocalDate2StringConverter implements Converter<LocalDate, String> {
		@Override
		public String convert(@NonNull LocalDate source) {
			return source.format(DateTimeFormatter.ofPattern("yyyy-MM-dd"));
		}
	}
}

这样参数就以正确的参数添加到Feign的URL中去传递了。当然,在FeignClient的接收方的RequestParam参数中,还需要添加

@DateTimeFormat(pattern = "yyyy-MM-dd HH:mm:ss")

注解代码才可以正确接收参数并反序列化成相应的时间对象。

在以Post等Body传参时,只需要配置Jackson相关的序列化配置就可以,因为Feign这个转成url的形式,默认没有走Jackson,而是MVC的String转换器。

你可能感兴趣的:(Spring,Boot,Spring,Cloud,Java)