Java easyExcel实现导入多sheet的Excel

 更新时间:2025年06月27日 11:15:50   作者:Uluoyu  
这篇文章主要为大家详细介绍了如何使用Java easyExcel实现导入多sheet的Excel,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下

1.官网

easyExcel官网

2.Excel样式

3.代码

@Slf4j
public class DynamicImportListener implements ReadListener<Map<Integer, String>> {

    /**
     * 从哪一行开始读数据
     */
    private final int headRowNumber;
    /**
     * 公司信息列
     */
    private final int companyInfoNumber;
    /**
     * 数据列
     */
    private final int headNumber;
    private final Map<Integer, Map<Integer, String>> rawRowsMap = new HashMap<>();
    private final List<CellExtra> extraMergeInfoList = new ArrayList<>();
    private final Map<Integer, String> headerMap = new LinkedHashMap<>();
    private final NavigableMap<Integer, String> companyInfoMap = new TreeMap<>();

    public DynamicImportListener(int headRowNumber, int companyInfoNumber, int headNumber) {
        this.headRowNumber = headRowNumber;
        this.companyInfoNumber = companyInfoNumber;
        this.headNumber = headNumber;
    }

    @Override
    public void invoke(Map<Integer, String> rowMap, AnalysisContext context) {
        int rowIndex = context.readRowHolder().getRowIndex();

        if (rowIndex == companyInfoNumber) {
            String company = rowMap.get(0);
            if (StrUtil.isNotBlank(company)) {
                companyInfoMap.put(rowIndex, company.trim());
            }
        } else if (rowIndex == headNumber) {
            for (Map.Entry<Integer, String> e : rowMap.entrySet()) {
                String v = e.getValue();
                if (StrUtil.isNotBlank(v)) {
                    headerMap.put(e.getKey(), v.trim());
                }
            }
        }else {
            rawRowsMap.put(rowIndex, rowMap);
        }
    }

    @Override
    public void extra(CellExtra extra, AnalysisContext context) {
        if (extra.getType() == CellExtraTypeEnum.MERGE
                && extra.getFirstRowIndex() >= headRowNumber - 1) {
            extraMergeInfoList.add(extra);
        }
    }

    @Override
    public void doAfterAllAnalysed(AnalysisContext context) {
        log.info("读取完毕:数据总行 {},表头列 {},合并单元格 {} 条",
                rawRowsMap.size(), headerMap.size(), extraMergeInfoList.size());
    }

    /**
     * 获取首条公司信息
     */
    public String getCompanyInfo() {
        return companyInfoMap.isEmpty() ? null : companyInfoMap.firstEntry().getValue();
    }

    /**
     * 获取所有合并单元格元数据
     */
    public List<CellExtra> getMergedRegions() {
        return Collections.unmodifiableList(extraMergeInfoList);
    }

    /**
     * 回填合并单元格数据
     */
    public void fillMergedCells() {
        if (extraMergeInfoList.isEmpty()) return;
        for (CellExtra extra : extraMergeInfoList) {
            int r1 = extra.getFirstRowIndex();
            int r2 = extra.getLastRowIndex();
            int c1 = extra.getFirstColumnIndex();
            String init = rawRowsMap.get(r1).get(c1);
            for (int rr = r1; rr <= r2; rr++) {
                Map<Integer, String> row = rawRowsMap.get(rr);
                if (row == null) continue;
                for (int cc = c1; cc <= extra.getLastColumnIndex(); cc++) {
                    row.put(cc, init);
                }
            }
        }
    }

    /**
     * 构建 VO 列表,固定字段 + extraFields
     */
    public <T> List<T> buildVoList(Class<T> voClass) {
        List<T> result = new ArrayList<>();
        int headerIdx = headNumber;
        int maxRow = rawRowsMap.keySet().stream().max(Integer::compareTo).orElse(headerIdx);

        for (int idx = headerIdx + 1; idx <= maxRow; idx++) {
            Map<Integer, String> rowMap = rawRowsMap.get(idx);
            if (rowMap == null) continue;
            try {
                T vo = voClass.getDeclaredConstructor().newInstance();
                // 填充列
                for (Map.Entry<Integer, String> head : headerMap.entrySet()) {
                    String headerName = head.getValue();
                    String cellVal = rowMap.get(head.getKey());
                    String value = (cellVal == null) ? "" : cellVal.trim();
                    boolean matched = false;
                    for (Field f : voClass.getDeclaredFields()) {
                        ExcelProperty prop = f.getAnnotation(ExcelProperty.class);
                        if (prop != null && Arrays.asList(prop.value()).contains(headerName)) {
                            f.setAccessible(true);
                            f.set(vo, convertType(f.getType(), value));
                            matched = true;
                            break;
                        }
                    }
                    if (!matched) {
                        Method m = voClass.getMethod("getExtraFields");
                        @SuppressWarnings("unchecked")
                        Map<String, String> extra = (Map<String, String>) m.invoke(vo);
                        extra.put(headerName, value);
                    }
                }

                result.add(vo);
            } catch (Exception e) {
                log.error("行 {} 构建 VO 失败: {}", idx + 1, e.getMessage());
            }
        }
        return result;
    }

    /**
     * @param targetType 目标类型
     * @param text 文本
     * @return java.lang.Object
     * @description 数据类型转换
     * @author zhaohuaqing
     * @date 2025/6/26 11:42
     */
    private Object convertType(Class<?> targetType, String text) {
        if (text == null) {
            return null;
        }
        String trimmed = text.trim();

        // 字符串
        if (targetType == String.class) {
            return trimmed;
        }
        // 原生数字类型
        if (targetType == Integer.class || targetType == int.class) {
            return Integer.valueOf(trimmed);
        }
        if (targetType == Long.class    || targetType == long.class) {
            return Long.valueOf(trimmed);
        }
        if (targetType == Double.class  || targetType == double.class) {
            return Double.valueOf(trimmed);
        }
        if (targetType == Float.class   || targetType == float.class) {
            return Float.valueOf(trimmed);
        }
        if (targetType == Short.class   || targetType == short.class) {
            return Short.valueOf(trimmed);
        }
        if (targetType == Byte.class    || targetType == byte.class) {
            return Byte.valueOf(trimmed);
        }
        // BigDecimal
        if (targetType == BigDecimal.class) {
            return new BigDecimal(trimmed);
        }
        // 布尔
        if (targetType == Boolean.class || targetType == boolean.class) {
            // 支持 "true"/"false",也支持 "1"/"0"
            if ("1".equals(trimmed) || "0".equals(trimmed)) {
                return "1".equals(trimmed);
            }
            return Boolean.valueOf(trimmed);
        }
        // Java 8 日期时间
        if (targetType == LocalDate.class) {
            // 默认 ISO 格式,或自定义
            return LocalDate.parse(trimmed, DateTimeFormatter.ISO_LOCAL_DATE);
        }
        if (targetType == LocalTime.class) {
            return LocalTime.parse(trimmed, DateTimeFormatter.ISO_LOCAL_TIME);
        }
        if (targetType == LocalDateTime.class) {
            return LocalDateTime.parse(trimmed, DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));
        }
        if (targetType == OffsetDateTime.class) {
            return OffsetDateTime.parse(trimmed, DateTimeFormatter.ISO_OFFSET_DATE_TIME);
        }
        if (targetType == ZonedDateTime.class) {
            return ZonedDateTime.parse(trimmed, DateTimeFormatter.ISO_ZONED_DATE_TIME);
        }
        // 旧版 java.util.Date
        if (targetType == java.util.Date.class) {
            try {
                // 你可以根据 Excel 导出格式,调整 SimpleDateFormat
                return new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").parse(trimmed);
            } catch (ParseException e) {
                throw new RuntimeException("日期解析失败: " + trimmed, e);
            }
        }
        return trimmed;
    }
}

如何使用

@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
@Accessors(chain = false) // 设置 chain = false,避免excel导入有问题
public class SopExtractMdsaVO {

   
    @ExcelProperty(value = "一级xxx")
    private String firstLevel;

  
    @ExcelProperty(value = "二级xxx")
    private String secondLevel;


    @ExcelProperty(value = "三级xxx")
    private String thirdLevel;
    
    @ExcelProperty(value = "xxx")
    private String formula;
    
    @ExcelProperty(value = "xxx")
    private String factor;
    
    @ExcelProperty(value = "xxx")
    private String referenceValue;
    
    @ExcelProperty(value = "xxx")
    private String element;
    
    @ExcelProperty(value = "xxx")
    private String scheme;
    
    @ExcelProperty(value = "超链接1")
    private String hyperlink1;
    
    @ExcelProperty(value = "超链接2")
    private String hyperlink2;
    
    @ExcelProperty(value = "超链接3")
    private String hyperlink3;

    /**
     * 动态列:所有未在 VO 明确定义的列
     */
    private Map<String, String> extraFields = new LinkedHashMap<>();

    /**
     * 管控方案IDs
     */
    private String controlPlanIds;

    /**
     * 输出物IDs
     */
    private String outputMaterialIds;

    /**
     * @return 非空的 hyperlink 列表
     */
    public List<String> nonBlankHyperlinks() {
        List<String> list = new ArrayList<>(10);
        if (StrUtil.isNotBlank(hyperlink1)) list.add(hyperlink1);
        if (StrUtil.isNotBlank(hyperlink2)) list.add(hyperlink2);
        if (StrUtil.isNotBlank(hyperlink3)) list.add(hyperlink3);
        return list;
    }
}
        InputStream inputStream = file.getInputStream();
        int headRowNumber = 0;  // 表头在 Excel 的第 2 行(从 1 开始计)
        DynamicImportListener listener = new DynamicImportListener(headRowNumber, 0, 1);

        // 1) 读数据、收集表头 & 合并单元格 & 公司信息
        EasyExcel.read(inputStream, listener)
                .extraRead(CellExtraTypeEnum.MERGE)
                .sheet("TEST")
                .headRowNumber(headRowNumber)
                .doRead();

        // 2) 外部拿公司信息
        String company = listener.getCompanyInfo();
        // 3) 回填合并单元格
        listener.fillMergedCells();
        // 4) 构建 VO 列表(包含固定字段 + extraFields)
        List<SopExtractMdsaVO> rows = listener.buildVoList(SopExtractMdsaVO.class);

到此这篇关于Java easyExcel实现导入多sheet的Excel的文章就介绍到这了,更多相关Java easyExcel导入Excel内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java JDBC连接数据库常见操作总结

    Java JDBC连接数据库常见操作总结

    这篇文章主要介绍了Java JDBC连接数据库常见操作,结合实例形式总结分析了java基于jdbc连接mysql、Oracle数据库及连接池相关操作技巧,需要的朋友可以参考下
    2019-03-03
  • java常用工具类 XML工具类、数据验证工具类

    java常用工具类 XML工具类、数据验证工具类

    这篇文章主要为大家详细介绍了java常用工具类,包括XML工具类、数据验证工具类,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-05-05
  • Maven依赖爆红的几种解决思路

    Maven依赖爆红的几种解决思路

    本文介绍了多种解决Maven依赖爆红的方法,包括删除.lastupdate文件、更改镜像设置、配置私服、删除错误依赖、手动修改依赖和检查pom文件错误等,通过这些方法可以有效解决Maven项目中遇到的依赖问题,感兴趣的可以了解一下
    2024-10-10
  • MybatisPlus QueryWrapper常用方法实例

    MybatisPlus QueryWrapper常用方法实例

    MyBatis-Plus(opens new window)是一个MyBatis(opens new window)的增强工具,在 MyBatis的基础上只做增强不做改变,为简化开发、提高效率而生,下面这篇文章主要给大家介绍了关于MybatisPlus QueryWrapper常用方法的相关资料,需要的朋友可以参考下
    2022-04-04
  • java使用servlet实现验证码

    java使用servlet实现验证码

    这篇文章主要介绍了java使用servlet实现验证码,简单实用,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • Springboot跨域CORS处理实现原理

    Springboot跨域CORS处理实现原理

    这篇文章主要介绍了Springboot跨域CORS处理实现原理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • 浅谈JSP与Servlet传值及对比(总结)

    浅谈JSP与Servlet传值及对比(总结)

    下面小编就为大家带来一篇浅谈JSP与Servlet传值及对比(总结)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • java搭建一个Socket服务器响应多用户访问

    java搭建一个Socket服务器响应多用户访问

    本篇文章主要介绍了java搭建一个Socket服务器响应多用户访问,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-02-02
  • 浅谈SpringMVC HandlerInterceptor诡异问题排查

    浅谈SpringMVC HandlerInterceptor诡异问题排查

    这篇文章主要介绍了浅谈SpringMVC HandlerInterceptor诡异问题排查,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-05-05
  • SpringCloud之注册中心之Nacos负载均衡详解

    SpringCloud之注册中心之Nacos负载均衡详解

    Nacos提供多种负载均衡策略,包括权重、同机房、同地域、同环境等,服务下线和权重配置可以通过Nacos管理界面进行,同时,Nacos使用Raft算法选举Leader节点,若IP地址改变可能会影响Leader选举,配置同集群优先访问可以提高访问速度,通过配置集群名称和负载均衡策略
    2025-03-03

最新评论