springboot 原生java_SpringBoot整合原生OpenFegin的坑(非SpringCloud)

本文介绍了在SpringBoot项目中整合原生OpenFeign时遇到的LocalDateTime字段反序列化异常问题。通过分析问题原因,指出了解决方案,即引入jackson-datatype-jsr310依赖并在LocalDateTime字段上添加JsonDeserialize注解,使用LocalDateTimeDeserializer进行反序列化。
摘要由CSDN通过智能技术生成

dc25a6aed567a1f78fbd189afc43525a.png

SpringBoot整合原生OpenFegin的坑(非SpringCloud)

96614dc726998c8944ad940ede2de203.png

项目集成OpenFegin

集成OpenFegin依赖

首先,我先跟大家说下项目的配置,整体项目使用的SpringBoot版本为2.2.6,原生的OpenFegin使用的是11.0,我们通过如下方式在pom.xml中引入OpenFegin。

UTF-8

false

1.8

11.0

io.github.openfeign

feign-core

${openfegin.version}

io.github.openfeign

feign-jackson

${openfegin.version}

这里,我省略了一些其他的配置项。

接下来,我就开始在我的项目中使用OpenFegin调用远程服务了。具体步骤如下。

实现远程调用

首先,创建OpenFeignConfig类,配置OpenFegin默认使用的Contract。

@Configuration

public class OpenFeignConfig{

@Bean

public Contract useFeignAnnotations(){

return new Contract.Default();

}

}

接下来,我们写一个通用的获取OpenFeign客户端的工厂类,这个类也比较简单,本质上就是以一个HashMap来缓存所有的FeginClient,这个的FeginClient本质上就是我们自定义的Fegin接口,缓存中的Key为请求连接的基础URL,缓存的Value就是我们定义的FeginClient接口。

public class FeginClientFactory{

/**

* 缓存所有的Fegin客户端

*/

private volatile static Map feginClientCache = new HashMap<>();

/**

* 从Map中获取数据

* @return

*/

@SuppressWarnings("unchecked")

public static T getFeginClient(Class clazz, String baseUrl){

if(!feginClientCache.containsKey(baseUrl)) {

synchronized (FeginClientFactory.class) {

if(!feginClientCache.containsKey(baseUrl)) {

T feginClient = Feign.builder().decoder(new JacksonDecoder()).encoder(new JacksonEncoder()).target(clazz, baseUrl);

feginClientCache.put(baseUrl, feginClient);

}

}

}

return (T)feginClientCache.get(baseUrl);

}

}

接下来,我们就定义一个FeginClient接口。

public interface FeginClientProxy{

@Headers("Content-Type:application/json;charset=UTF-8")

@RequestLine("POST /user/login")

UserLoginVo login(UserLoginVo loginVo);

}

接下来,我们创建SpringBoot的测试类。

@RunWith(SpringRunner.class)

@SpringBootTest

public class IcpsWeightStarterTest{

@Test

public void testUserLogin(){

ResponseMessage result = FeginClientFactory.getFeginClient(FeginClientProxy.class, "http://127.0.0.1").login(new UserLoginVo("zhangsan", "123456", 1));

System.out.println(JsonUtils.bean2Json(result));

}

}

一切准备就绪,运行测试。麻蛋,出问题了。主要的问题就是通过OpenFeign请求返回值LocalDateTime字段会发生异常!!!

注:此时异常时,我们在LocalDateTime字段上添加的注解如下所示。

import java.time.LocalDateTime;

import com.baomidou.mybatisplus.annotation.FieldFill;

import com.baomidou.mybatisplus.annotation.TableField;

import com.fasterxml.jackson.annotation.JsonFormat;

@TableField(value = "CREATE_TIME", fill = FieldFill.INSERT)

@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", locale = "zh", timezone = "GMT+8")

private LocalDateTime createTime;

解决问题

问题描述

SpringBoot通过原生OpenFeign客户端调用HTTP接口,如果返回值中包含LocalDateTime类型(包括其他JSR-310中java.time包的时间类),在客户端可能会出现反序列化失败的错误。错误信息如下:

Caused by:com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance of `java.time.LocalDateTime` (no Creators, like default construct, exist): no String-argument constructor/factory method to deserialize from String value ('2020-10-07T11:04:32')

问题分析

从客户端调用fegin,也是相当于URL传参就相当于经过一次JSON转换,数据库取出‘2020-10-07T11:04:32’数据这时是时间类型,进过JSON之后就变成了String类型,T就变成了字符不再是一个特殊字符,因此String的字符串“2020-10-07T11:04:32”反序列化就会失败。

问题解决

在项目中增加依赖。

com.fasterxml.jackson.datatype

jackson-datatype-jsr310

2.9.9

注:如果是用的是SpringBoot,并且明确指定了SpringBoot版本,引入jackson-datatype-jsr310时,可以不用指定版本号。

接下来,在POJO类的LocalDateTime类型字段增加如下注解。

import com.fasterxml.jackson.databind.annotation.JsonDeserialize;

import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;

添加后的效果如下所示。

import java.time.LocalDateTime;

import com.baomidou.mybatisplus.annotation.FieldFill;

import com.baomidou.mybatisplus.annotation.TableField;

import com.fasterxml.jackson.annotation.JsonFormat;

import com.fasterxml.jackson.databind.annotation.JsonDeserialize;

import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer;

@TableField(value = "CREATE_TIME", fill = FieldFill.INSERT)

@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", locale = "zh", timezone = "GMT+8")

@JsonDeserialize(using = LocalDateTimeDeserializer.class)

private LocalDateTime createTime;

此时,再次调用远程接口,问题解决。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值