EasyExcel 自定义通用转换器

问题描述

近期在工作过程中,业务涉及各种导入导出,且包含单选、多选问题,针对于easyExcel的一些枚举转换数据,有了一点其他想法,按照官方文档中:读Excel | Easy Excel (alibaba.com),提供了自定义转换器的实现接口,但是我在网上搜索发现都比较单一,也就是一个枚举类需要写一个单独的转换器,这样的话会增加无意义工作量,所以想到了通过自定义注解+重新easyExcel转换器实现通用的easyExcle转换器。

话不多说,直接上代码:

1.重写EasyExcel转换器接口,对excel读和写方法都进行自定义

package com.example.test.excel;

import com.alibaba.excel.converters.Converter;
import com.alibaba.excel.enums.CellDataTypeEnum;
import com.alibaba.excel.metadata.GlobalConfiguration;
import com.alibaba.excel.metadata.data.ReadCellData;
import com.alibaba.excel.metadata.data.WriteCellData;
import com.alibaba.excel.metadata.property.ExcelContentProperty;
import org.springframework.util.StringUtils;

import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

/**
 * excel导入导出针对枚举类型的转换器
 *
 * @Author 996
 * @Date 2024/3/13
 */
public class EasyExcelConvert implements Converter<Object> {
    /**
     * 枚举列表
     */
    private Map<String, String> enumMap = new HashMap<>();


    /**
     * excel转化后的类型
     *
     * @return
     */
    @Override
    public Class<?> supportJavaTypeKey() {
        return Object.class;
    }

    /**
     * excel中的数据类型,统一设置字符串
     *
     * @return
     */
    @Override
    public CellDataTypeEnum supportExcelTypeKey() {
        return CellDataTypeEnum.STRING;
    }

    /**
     * 导入转换
     * @param cellData            当前单元格对象
     * @param contentProperty     当前单元格属性
     * @param globalConfiguration
     * @return
     * @throws Exception
     */
    @Override
    public Object convertToJavaData(ReadCellData<?> cellData, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) throws Exception {
        String cellMsg = cellData.getStringValue();
        Field field = contentProperty.getField();
        EnumFiledConvert enumFiledConvert = field.getAnnotation(EnumFiledConvert.class);
        if (enumFiledConvert == null) {
            return null;
        }
        String enumStr = enumFiledConvert.enumMap();
        // 解析枚举映射关系
        getEnumMap(enumStr, true);
        // 是否为单选
        boolean single = enumFiledConvert.single();
        // 如果是单选,默认Java属性为integer
        if (single) {
            String res = enumMap.get(cellMsg);
            return StringUtils.hasText(res) ? Integer.valueOf(res) : null;
        } else {
            // 多选分隔符
            String spiteChar = enumFiledConvert.spiteChar();
            // 多选枚举,默认Java属性为字符串,格式为 key1,key2,key3
            List<String> strStr = Arrays.asList(cellMsg.split(spiteChar)).stream().map(s -> String.valueOf(enumMap.get(s))).collect(Collectors.toList());
            String str = String.join(spiteChar, strStr);
            return str;
        }
    }

    /**
     * 导出转化
     * @param value               当前值
     * @param contentProperty     当前单元格属性
     * @param globalConfiguration
     * @return
     * @throws Exception
     */
    @Override
    public WriteCellData<?> convertToExcelData(Object value, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) throws Exception {
        Field field = contentProperty.getField();
        EnumFiledConvert enumFiledConvert = field.getAnnotation(EnumFiledConvert.class);
        if (enumFiledConvert == null) {
            return new WriteCellData();
        }
        // 解析枚举字符串
        String enumStr = enumFiledConvert.enumMap();
        getEnumMap(enumStr, false);
        // 是否为单选
        boolean single = enumFiledConvert.single();
        // 如果是单选,默认Java属性为integer
        if (single) {
            return new WriteCellData(enumMap.getOrDefault(String.valueOf(value), ""));
        } else {
            // 多选分隔符
            String spiteChar = enumFiledConvert.spiteChar();
            // 多选枚举,默认Java属性为字符串,格式为 key1,key2,key3
            List<String> strStr = Arrays.asList(String.valueOf(value).split(spiteChar)).stream().map(s -> String.valueOf(enumMap.get(s))).collect(Collectors.toList());
            String str = String.join(spiteChar, strStr);
            return new WriteCellData(str);
        }
    }

    /**
     * 根据注解配置的枚举映射字符串进行解析到map中
     * @param mapStr
     * @param readOrWrite 读excel 、 写excel
     */
    private void getEnumMap(String mapStr, boolean readOrWrite) {
        String[] enumS = mapStr.split(",");
        for (String anEnum : enumS) {
            String[] data = anEnum.split("-");
            if (readOrWrite) {
                // 读excel excel中的数据都是value,转换成key
                enumMap.put(data[1], data[0]);
            } else {
                // 写excel  Java中的数据都是key,转换成value
                enumMap.put(data[0], data[1]);
            }
        }
    }
}

2.自定义注解,包含3个参数,分别是枚举类型的字符串,多选场景下的分隔符,单选枚举or多选枚举

package com.example.test.excel;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 对excel导入时,处理枚举转化
 * @Author 996
 * @Date 2024/3/13
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface EnumFiledConvert {


    /**
     * 枚举映射map  key-value,key-value,key-value,key-value
     * @return
     */
    String enumMap() default "";

    /**
     * 枚举类导入、导出在excel中的分隔符号
     * @return
     */
    String spiteChar() default ",";

    /**
     * 单选 or 多选
     * @return
     */
    boolean single() default true;
}

        很简单吧?就是通过自定义注解,把枚举类映射关系传递到注解的参数里面。目前这个实现方式是写死在代码里面的。如果说枚举映射都存在数据库中的话,可以对方法进行改造,枚举参数传递成数据库里面的分类标识,在转换器里面去获取数据库里对应的枚举内容。

上样例!

先写一个用户类

import com.alibaba.excel.annotation.ExcelProperty;
import com.example.test.excel.EasyExcelConvert;
import com.example.test.excel.EnumFiledConvert;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

/**
 * @Author 996
 * @Date 2024/3/20
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
public class ExcelDto {
    @ExcelProperty(value = "姓名")
    private String name;

    @ExcelProperty(value = "性别",converter = EasyExcelConvert.class)
    @EnumFiledConvert(enumMap = "0-保密,1-男,2-女")
    private Integer sex;

    @ExcelProperty(value = "爱好",converter = EasyExcelConvert.class)
    @EnumFiledConvert(enumMap = "1-篮球,2-足球,3-乒乓球,4-羽毛球",single = false)
    private String hobbies;
}

建一个对象列表,模拟数据库查询出来的数据,测试导出是否成功转换。(该方法采用excel自动生成excel,也可以是使用fill填充的方式写入指定excel模板

调用接口,查看Excel导出效果,一切正常:

再测试导入,直接将这个Excel提交,查看后台接收效果:

整体代码都比较简单,仅仅是提供一种思路去尽可能减少这种处理枚举转换上的无意义工作量,如果各位大佬有更好的意见,还请不吝赐教。

  • 18
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
对于 EasyExcel 来说,自定义转换器是用来处理 Excel 中的自定义对象转换的。下面是一个示例代码,演示了如何实现自定义转换器: ```java // 自定义转换器 public class CustomConverter implements Converter<CustomObject> { @Override public Class<CustomObject> supportJavaTypeKey() { return CustomObject.class; } @Override public CellData<?> convertToExcelData(CustomObject customObject, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) { // 将自定义对象转换为需要导出的数据类型,比如字符串 String data = customObject.toString(); return new CellData<>(data); } @Override public CustomObject convertToJavaData(CellData<?> cellData, ExcelContentProperty contentProperty, GlobalConfiguration globalConfiguration) { // 将读取到的数据转换为自定义对象 String data = Objects.requireNonNull(cellData.getStringValue()); // 这里根据实际情况进行解析,创建自定义类对象 CustomObject customObject = new CustomObject(data); return customObject; } } ``` 在使用 EasyExcel 导出或导入 Excel 文件时,可以注册自定义转换器,示例代码如下: ```java public class ExcelUtils { public static void exportData(List<CustomObject> dataList, String filePath) { // 创建工作簿 try (OutputStream outputStream = new FileOutputStream(filePath); ExcelWriter excelWriter = EasyExcel.write(outputStream).build()) { // 注册自定义转换器 ConverterRegistry converterRegistry = excelWriter.getConverterRegistry(); converterRegistry.registerConverter(new CustomConverter()); // 设置表头和数据 Sheet sheet = new Sheet(1, 0, CustomObject.class); excelWriter.write(dataList, sheet); } catch (IOException e) { e.printStackTrace(); } } public static List<CustomObject> importData(String filePath) { // 读取Excel文件 try (InputStream inputStream = new FileInputStream(filePath); ExcelReader excelReader = EasyExcel.read(inputStream).build()) { // 注册自定义转换器 ConverterRegistry converterRegistry = excelReader.getConverterRegistry(); converterRegistry.registerConverter(new CustomConverter()); // 设置表头和数据 Sheet sheet = new Sheet(1, 1, CustomObject.class); ReadSheet readSheet = EasyExcel.readSheet(0).build(); excelReader.read(readSheet); // 获取读取到的数据 List<CustomObject> dataList = readSheet.getList(); return dataList; } catch (IOException e) { e.printStackTrace(); } return Collections.emptyList(); } } ``` 以上代码中的 `CustomObject` 是自定义的类,你需要根据实际情况进行替换。同时,你可以根据需要自定义其他类型的转换器来处理不同类型的数据。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值