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

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/ttzommed/article/details/89188619

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

问题来源

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

源码追踪

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

解决方式

最快的解决方式即是直接配置相关的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转换器。

猜你喜欢

转载自blog.csdn.net/ttzommed/article/details/89188619
今日推荐