springboot 枚举序列化时,动态添加翻译值

springboot 枚举序列化时,动态添加翻译值

前提:了解springboot与FastJson的整合以及相关的全局配置

一、需求
  • 后端返回给前端值,添加枚举值翻译,如:sex=1,本来的json是{"sex":1},期望是:{"sex":1,"sexName":"男"}
二、分析
  • 在json序列化时,把序列化后的字符串加上xxxName属性,根据FastJson的用法,可以通过继承AfterFilter来实现
三、实现思路
  • 定义转换过滤器,继承 com.alibaba.fastjson2.filter.AfterFilter,重写方法writeAfter方法

  • 自定义注解@EnumDescribevalue=将要注入的枚举类)

@Retention(RetentionPolicy.RUNTIME)
public @interface EnumDescribe {

    /**
     * 继承EnumName接口的枚举类
     *
     * @return {@link Class}<{@link ?} {@link extends} {@link IEnumDescribe}>
     */
    Class<? extends IEnumDescribe<?>> value();

    /**
     * 前缀
     *
     * @return {@link String}
     */
    String prefix() default "";

    /**
     * 后缀
     *
     * @return {@link String}
     */
    String suffix() default "Name";
}
  • 如果序列的类的包名前缀是:com.xxxxx,那么就进行处理

  • 根据当前序列化的类,获取其所有属性

  • 判断属性是否有注解@EnumDescribe

  • 如果有注解,那么获取的注解的value枚举类

  • 根据当前类的当前属性值与枚举类中各个元素的值进行比较

  • 如果相等,则为当前序列化对象写入(key,value) (key=枚举元素属性名+Name value=枚举元素的描述值·)

  • 以上代码描述:

public class EnumConvertAfterFilter extends AfterFilter {

    @Override
    public void writeAfter(Object object) {
        Class<?> objectClass = object.getClass();
        if (objectClass.getName().startsWith("com.xxxx") && ClassUtil.isNormalClass(objectClass)) {
            Field[] declaredFields = objectClass.getDeclaredFields();
            for (Field field : declaredFields) {
                EnumDescribe annotation = field.getAnnotation(EnumDescribe.class);
                if (annotation != null) {
                    try {
                        field.setAccessible(true);
                        Class<? extends IEnumDescribe<?>> clazz = annotation.value();
                        IEnumDescribe<?>[] enumConstants = clazz.getEnumConstants();
                        for (IEnumDescribe<?> enumConstant : enumConstants) {
                            if (enumConstant.getValue().equals(field.get(object))) {
                                String key = field.getName().concat(annotation.suffix());
                                if (StringUtils.hasText(annotation.prefix())) {
                                    key = StrUtil.upperFirstAndAddPre(key, annotation.prefix());
                                }
                                super.writeKeyValue(key, enumConstant.getDescription());
                                break;
                            }
                        }
                    } catch (IllegalAccessException e) {
                        throw new RuntimeException(e);
                    }
                }
            }
        }
    }
}
  • 把转换过滤器放入FastJsonConfig的配置中,然后放在入转换器中,自定义转换器
    @Bean
    @ConditionalOnMissingBean(FastJsonHttpMessageConverter.class)
    public FastJsonHttpMessageConverter converter() {
        FastJsonHttpMessageConverter converter = new FastJsonHttpMessageConverter();
        FastJsonConfig config = new FastJsonConfig();
        // 重点
        config.setWriterFilters(new EnumConvertAfterFilter());
        converter.setFastJsonConfig(config);
        return converter;
    }
四、用法
    @EnumDescribe(EnumSex.class)
    private Integer sex;
五、相关枚举类需要实现IEnumDescribe
public interface IEnumDescribe<T> {

    /**
     * 获取id
     *
     * @return {@link Integer}
     */
    T getValue();

    /**
     * 获取名字
     *
     * @return {@link String}
     */
    String getDescription();
}

注:

  • 相关FastJson文档:https://www.w3cschool.cn/fastJson/FastJson-aiserializefilter.html
  • 相关springboot与FastJson整合文档:https://blog.csdn.net/lrb0677/article/details/125287043

如有描述不清楚的地方,欢迎留言,我会及时回复

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
枚举类型是一个有限的取集合,可以通过序列化和反序列化将其转换为字符串或数字。 序列化:将枚举类型转换为字符串或数字,以便于存储或传输。在C#中,可以使用Enum.ToString()方法将枚举转换为字符串,也可以使用(Enum)枚举枚举转换为整数。 反序列化:将字符串或数字转换为枚举类型。在C#中,可以使用Enum.Parse()方法将字符串转换为枚举类型,也可以使用(Enum)整数将整数转换为枚举类型。 以下是一个示例代码: ```csharp using System; enum DayOfWeek { Sunday, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday } class Program { static void Main(string[] args) { // 序列化 DayOfWeek day = DayOfWeek.Monday; string dayStr = day.ToString(); int dayInt = (int)day; Console.WriteLine($"dayStr: {dayStr}"); Console.WriteLine($"dayInt: {dayInt}"); // 反序列化 string dayStr2 = "Wednesday"; DayOfWeek day2 = (DayOfWeek)Enum.Parse(typeof(DayOfWeek), dayStr2); int dayInt2 = 5; DayOfWeek day3 = (DayOfWeek)dayInt2; Console.WriteLine($"day2: {day2}"); Console.WriteLine($"day3: {day3}"); } } ``` 输出结果: ``` dayStr: Monday dayInt: 1 day2: Wednesday day3: Friday ``` 在上面的示例中,我们首先将枚举类型`DayOfWeek`的序列化为字符串和整数,然后将其反序列化枚举类型的。注意,反序列化需要使用`Enum.Parse()`方法将字符串转换为枚举类型,或者使用强制类型转换将整数转换为枚举类型。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值