从零搭建开发脚手架 SpringMvc中枚举类型参数的序列化和反序列化

背景

以下是我们一般的枚举类配合SpringMvc使用方式。

枚举类如下

public enum Distance {
    KILOMETER("km", 1000),
    MILE("miles", 1609.34),
    METER("meters", 1);

    private String unit;
    private final double meters;

    Distance(String unit, double meters) {
        this.unit = unit;
        this.meters = meters;
    }

实体类如下

public class City {
    private String id;
    private Distance distance;

Controller如下

-- /api/v1?distance=KILOMETER
@GetMapping
public void get(Distance distance) 
    
-- /api/v1?distance=KILOMETER&id=1
@GetMapping
public void get(City city) 
    
-- json
{
  "id": "1",
  "distance": "KILOMETER"
}
@PostMapping
public void post(@RequestBody City city)
    
-- 反序列化    
@GetMapping
public City get()
-- json    
{
  "id": "1",
  "distance": "KILOMETER"
}    

默认所有的参数和响应都是大写的字符串枚举名称

优化

我们想自定义序列化和反序列化字段,例如使用unit或者meters属性。

针对普通请求参数

-- /api/v1?distance=KILOMETER
@GetMapping
public void get(Distance distance) 

-- /api/v1?distance=KILOMETER&id=1
@GetMapping
public void get(City city) 

1. 定义IEnum接口

public interface IEnum<T extends Serializable> {
    T getValue();
}

2.基于SpringMvc的Converter实现StringToEnum转换

public class StringToEnumConvertFactory implements ConditionalGenericConverter {
    @Override
    public boolean matches(TypeDescriptor sourceType, TypeDescriptor targetType) {
        return IEnum.class.isAssignableFrom(targetType.getObjectType()) && sourceType.getObjectType() == String.class;
    }
    @Override
    public Object convert(Object source, TypeDescriptor sourceType, TypeDescriptor targetType) {
        IEnum[] enums = (IEnum[]) targetType.getObjectType().getEnumConstants();
        for (IEnum anEnum : enums) {
            if (Objects.equals(anEnum.getValue(), source)) {
                return anEnum;
            }
        }
        return null;
    }
}

3.把转换器注册到SpringMvc

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
    @Override
    public void addFormatters(FormatterRegistry registry) {
        registry.addConverter(new StringToEnumConvertFactory());
    }    

4.自定义枚举实现定义IEnum接口

public enum Distance implements IEnum<String> {
    KILOMETER("km", 1000),
    MILE("miles", 1609.34),
    METER("meters", 1);
    private String unit;
    private final double meters;

    Distance(String unit, double meters) {
        this.unit = unit;
        this.meters = meters;
    }
    @Override // 这里选择转换的属性
    public String getValue() {
        return unit;
    }
}

实现效果如下

-- /api/v1?distance=km
-- /api/v1?distance=miles
@GetMapping
public void get(Distance distance) 

-- /api/v1?distance=km&id=1
@GetMapping
public void get(City city) 

针对RequestBody参数

-- json
{
  "id": "1",
  "distance": "KILOMETER"
}
@PostMapping
public void post(@RequestBody City city)

直接在枚举的指定get方法使用@JsonValue注解即可。

public enum Distance implements IEnum<String> {
    KILOMETER("km", 1000),
    MILE("miles", 1609.34),
    METER("meters", 1);
    private String unit;
    private final double meters;

    Distance(String unit, double meters) {
        this.unit = unit;
        this.meters = meters;
    }
    @Override
    @JsonValue // 这里配置
    public String getValue() {
        return unit;
    }
}

实现效果如下

-- json
{
  "id": "1",
  "distance": "km"
}
@PostMapping
public void post(@RequestBody City city)

针对枚举类型响应

直接在get方法使用@JsonValue注解即可。跟上面方式一样

实现效果如下

-- 反序列化    
@GetMapping
public City get()
-- json    
{
  "id": "1",
  "distance": "km"
}   

更多Jackson序列化和反序列化参考:https://blog.csdn.net/hj_5346/article/details/127269700

  • @JsonFormat(shape = JsonFormat.Shape.OBJECT)
  • @JsonValue
  • @JsonSerialize(using = DistanceSerializer.class)
  • @JsonProperty(“distance-in-km”)
  • @JsonCreator
  • @JsonDeserialize(using = CustomEnumDeserializer.class)
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
【资源介绍】 基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip基于SpringBoot+SpringMVC和多种组件构建的企业信息化开发基础平台源码.zip 【备注】 该项目是个人毕设项目,答辩评审分达到95分,代码都经过调试测试,确保可以运行!欢迎下载使用,可用于小白学习、进阶。 该资源主要针对计算机、通信、人工智能、自动化等相关专业的学生、老师或从业者下载使用,亦可作为期末课程设计、课程大作业、毕业设计等。 项目整体具有较高的学习借鉴价!基础能力强的可以在此基础上修改调整,以实现不同的功能。 欢迎下载,欢迎沟通,互相学习,共同进步!提供答疑!
在Spring MVC整合RedisTemplate时,如果遇到序列化超时的问题,可以尝试以下几个解决方法: 1. 调整序列化方式:默认情况下,RedisTemplate使用JdkSerializationRedisSerializer作为默认的序列化方式。这种序列化方式可能导致序列化超时问题。可以尝试使用其他序列化方式,如Jackson2JsonRedisSerializer或StringRedisSerializer。 ```java // 使用Jackson2JsonRedisSerializer作为序列化方式 RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>(); redisTemplate.setConnectionFactory(jedisConnectionFactory); redisTemplate.setDefaultSerializer(new Jackson2JsonRedisSerializer<>(Object.class)); ``` 2. 增加超时配置:在配置连接工厂的时候,可以增加超时配置,设置合适的超时时间,确保在序列化过程不会超时。 ```xml <!-- Redis连接工厂配置 --> <bean id="jedisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"> <!-- Redis服务器地址 --> <property name="hostName" value="localhost" /> <!-- Redis服务器端口号 --> <property name="port" value="6379" /> <!-- Redis密码 --> <property name="password" value="password" /> <!-- Redis连接超时时间 --> <property name="timeout" value="5000" /> <!-- Redis连接池 --> <property name="poolConfig" ref="jedisPoolConfig" /> </bean> ``` 3. 检查对象是否可序列化:如果使用自定义的对象进行存储,需要确保这些对象实现了Serializable接口,以便进行正确的序列化。 ```java public class CustomObject implements Serializable { // ... } ``` 通过以上方法,可以尝试解决Spring MVCRedisTemplate序列化超时的问题。如果问题仍然存在,请提供更多详细信息以便我们能够更好地帮助你解决问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lakernote

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值