【问题解决】apache.poi 3.1.4版本升级到 5.2.3,导出文件报错版本无法解析

【问题解决】apache.poi 3.1.4版本升级到 5.2.3,导出文件报错无法解析

3.1.4版本代码:

bash 复制代码
    /**
     * 创建workbook
     * @param inp
     * @return
     * @throws Exception
     */
    public Workbook createworkbook(InputStream inp) throws Exception {


        if (!inp.markSupported()) {
            inp = new PushbackInputStream(inp, 8);
        }

        if (POIFSFileSystem.hasPOIFSHeader(inp)) {
            return new HSSFWorkbook(inp);
        }
        if (POIXMLDocument.hasOOXMLHeader(inp)) {
            return new XSSFWorkbook(OPCPackage.open(inp));
        }
        throw new IllegalArgumentException("你的excel版本目前poi解析不了");
    }

在POI 4.0.x版本过后,POIFSFileSystem.hasPOIFSHeader() POIXMLDocument.hasOOXMLHeader()便弃用了,所以我们需要找到新版本的替代方法,为此查阅了不少资料。

这里有两个方案可以参考一下:

java 复制代码
public Workbook createworkbook(InputStream inp) throws Exception {
    try (PushbackInputStream pushbackInputStream = new PushbackInputStream(inp, 8)) {
        byte[] header = new byte[8];
        int read = pushbackInputStream.read(header);
        pushbackInputStream.unread(header, 0, read);

        if (POIUtils.hasOOXMLHeader(header)) {
            return new XSSFWorkbook(OPCPackage.open(pushbackInputStream));
        } else if (POIUtils.hasPOIFSHeader(header)) {
            return new HSSFWorkbook(pushbackInputStream);
        } else {
            throw new IllegalArgumentException("不支持的 Excel 格式");
        }
    }
}

主要变更如下:

  1. 使用 POIUtils.hasOOXMLHeader(header) 方法替换 POIDataSamples.getSpreadSheetInstance().acceptsFile(header) 方法。这个方法可以检测 XSSF 格式(OOXML)。
  2. 使用 POIUtils.hasPOIFSHeader(header) 方法检测 HSSF 格式(97-2003 Excel)。
  3. 如果以上两种方式都无法识别,则抛出一个自定义的异常。

这种方式可以有效地替换之前的代码,并适用于 Apache POI 5.2.3 版本。

需要注意的是,需要引入 org.apache.poi.util.POIUtils 类,这个类是 Apache POI 5.2.3 中新增的一个工具类,提供了一些常用的工具方法。
2.

java 复制代码
public Workbook createworkbook(InputStream inp) throws Exception {
    try (PushbackInputStream pushbackInputStream = new PushbackInputStream(inp, 8)) {
        byte[] header = new byte[8];
        int read = pushbackInputStream.read(header);
        pushbackInputStream.unread(header, 0, read);

        if (isXSSF(header)) {
            return new XSSFWorkbook(OPCPackage.open(pushbackInputStream));
        } else if (isHSSF(header)) {
            return new HSSFWorkbook(pushbackInputStream);
        } else {
            throw new IllegalArgumentException("不支持的 Excel 格式");
        }
    }
}

private boolean isXSSF(byte[] header) {
    // 检查 OOXML 文件头标识
    return header[0] == (byte) 0x50 && header[1] == (byte) 0x4B && header[2] == (byte) 0x03 && header[3] == (byte) 0x04;
}

private boolean isHSSF(byte[] header) {
    // 检查 POIFS 文件头标识
    return header[0] == (byte) 0xD0 && header[1] == (byte) 0xCF && header[2] == (byte) 0x11 && header[3] == (byte) 0xE0;
}	

这个替代方案中,自己实现了 isXSSFisHSSF 方法来检测 OOXML 和 POIFS 文件头标识,达到了同样的效果。

踩坑啊!

相关推荐
better_liang25 分钟前
每日Java面试场景题知识点之-消息队列MQ核心场景与实战
java·面试·kafka·消息队列·rabbitmq·rocketmq·mq
小江的记录本37 分钟前
【JVM虚拟机】垃圾回收GC:四种引用类型:强引用、软引用、弱引用、虚引用(附《思维导图》+《面试高频考点清单》)
java·jvm·spring boot·后端·python·spring·面试
小马爱打代码1 小时前
Spring源码 第四篇:Spring 5 源码深度拆解:AOP 全流程核心原理
java·后端·spring
better_liang1 小时前
每日Java面试场景题知识点之-SpringBoot启动流程
java·面试·springboot·源码解析·启动流程
RyFit1 小时前
Java + AI 实战:Spring AI 从入门到企业级落地
java·人工智能·spring
ZhengEnCi3 小时前
01-如何监听接口调用情况?
java·spring boot·后端
JAVA面经实录9174 小时前
MyBatis学习体系
java·mybatis
java1234_小锋4 小时前
在 Spring AI 中如何实现函数调用(Function Calling)?请说明其基本原理和应用场景。
java·人工智能·spring
小马爱打代码4 小时前
Spring源码 第九篇:Spring 5 源码深度拆解 - Spring 事件驱动模型
java·后端·spring
ForgeAI码匠5 小时前
ForgeAdmin|Spring Boot 3 后台框架的自动配置设计:少写配置,多做组合
java·spring boot·后端