java.lang.NoClassDefFoundError: org/apache/poi/xssf/usermodel/XSSFWorkbook

java.lang.NoClassDefFoundError: org/apache/poi/xssf/usermodel/XSSFWorkbook

springboot集成poi,结果报了如下错误

java.lang.NoClassDefFoundError: org/apache/poi/xssf/usermodel/XSSFWorkbook
	at com.usoft.sschool_manage.util.excel.ExcelUtil.foodInformation(ExcelUtil.java:73) ~[classes/:na]
	at com.usoft.sschool_manage.service.UploadExcelUtil.uploadExcel(UploadExcelUtil.java:103) ~[classes/:na]
	at com.usoft.sschool_manage.controller.UploadExcelController.getExcel(UploadExcelController.java:29) ~[classes/:na]
	at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:1.8.0_172]
	at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) ~[na:1.8.0_172]
	at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) ~[na:1.8.0_172]
	at java.lang.reflect.Method.invoke(Method.java:498) ~[na:1.8.0_172]
	at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:189) ~[spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:138) ~[spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:102) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:892) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:797) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1038) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:942) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1005) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:908) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at javax.servlet.http.HttpServlet.service(HttpServlet.java:661) ~[servlet-api.jar:na]
	at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:882) ~[spring-webmvc-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at javax.servlet.http.HttpServlet.service(HttpServlet.java:742) ~[servlet-api.jar:na]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:231) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52) ~[tomcat-websocket.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:99) ~[spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:92) ~[spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.springframework.web.filter.HiddenHttpMethodFilter.doFilterInternal(HiddenHttpMethodFilter.java:93) ~[spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.springframework.boot.web.servlet.support.ErrorPageFilter.doFilter(ErrorPageFilter.java:130) [spring-boot-2.1.4.RELEASE.jar:2.1.4.RELEASE]
	at org.springframework.boot.web.servlet.support.ErrorPageFilter.access$000(ErrorPageFilter.java:66) [spring-boot-2.1.4.RELEASE.jar:2.1.4.RELEASE]
	at org.springframework.boot.web.servlet.support.ErrorPageFilter$1.doFilterInternal(ErrorPageFilter.java:105) [spring-boot-2.1.4.RELEASE.jar:2.1.4.RELEASE]
	at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.boot.web.servlet.support.ErrorPageFilter.doFilter(ErrorPageFilter.java:123) [spring-boot-2.1.4.RELEASE.jar:2.1.4.RELEASE]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:200) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107) [spring-web-5.1.6.RELEASE.jar:5.1.6.RELEASE]
	at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:193) [catalina.jar:8.5.35]
	at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:166) [catalina.jar:8.5.35]
	at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:198) [catalina.jar:8.5.35]
	at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96) [catalina.jar:8.5.35]
	at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:493) [catalina.jar:8.5.35]
	at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:140) [catalina.jar:8.5.35]
	at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:81) [catalina.jar:8.5.35]
	at org.apache.catalina.valves.AbstractAccessLogValve.invoke(AbstractAccessLogValve.java:650) [catalina.jar:8.5.35]
	at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:87) [catalina.jar:8.5.35]
	at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:342) [catalina.jar:8.5.35]
	at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:800) [tomcat-coyote.jar:8.5.35]
	at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:66) [tomcat-coyote.jar:8.5.35]
	at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:806) [tomcat-coyote.jar:8.5.35]
	at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1498) [tomcat-coyote.jar:8.5.35]
	at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49) [tomcat-coyote.jar:8.5.35]
	at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149) [na:1.8.0_172]
	at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624) [na:1.8.0_172]
	at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61) [tomcat-util.jar:8.5.35]
	at java.lang.Thread.run(Thread.java:748) [na:1.8.0_172]
Caused by: java.lang.ClassNotFoundException: org.apache.poi.xssf.usermodel.XSSFWorkbook
	at org.apache.catalina.loader.WebappClassLoaderBase.loadClass(WebappClassLoaderBase.java:1328) ~[catalina.jar:8.5.35]
	at org.apache.catalina.loader.WebappClassLoaderBase.loadClass(WebappClassLoaderBase.java:1156) ~[catalina.jar:8.5.35]
	... 65 common frames omitted

。。。。我擦,居然说我的类未定义,于是我查看的相关类:

package com.usoft.sschool_manage.util.excel;


import com.usoft.smartschool.pojo.XnFood;
import com.usoft.smartschool.pojo.XnScore;
import com.usoft.smartschool.util.MyResult;
import com.usoft.smartschool.util.ObjectUtil;
import org.apache.poi.ss.usermodel.CellType;
import org.apache.poi.xssf.usermodel.XSSFCell;
import org.apache.poi.xssf.usermodel.XSSFRow;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;

import java.io.File;
import java.io.FileInputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * Excel文件解析
 * @Author jijh
 * @Date 2019/5/17 11:18
 */

public class ExcelUtil {


    /**
     * 教师
     */
    private static final int TEACHER = 1;

    /**
     * 班级
     */
    private static final int CLASS = 2;

    /**
     * 课程
     */
    private static final int COURSE = 3;

    /**
     * 菜单
     */
    private static final int FOOD = 4;

    /**
     * 成绩
     */
    private static final int SCORE = 5;

    /**
     * 学生
     */
    private static final int STUDENT = 6;


    /**
     * 菜单导入
     *
     * @param schoolId
     * @param userId
     * @param file
     * @return
     */
    public static MyResult foodInformation(Integer schoolId, Integer userId, File file) {
        //workbook = null;
        try {
            FileInputStream fis = new FileInputStream(file);
            XSSFWorkbook workbook = new XSSFWorkbook(fis);
            XSSFSheet sheet = workbook.getSheetAt(0);
            int rowCount = sheet.getPhysicalNumberOfRows();
            List<XnFood> xnFoods = new ArrayList<>();
            for (int i = 1; i < rowCount; i++) {
                XSSFRow row = sheet.getRow(i);
                XnFood xnFood = new XnFood();
                xnFood.setSid(schoolId);
                for (int j = 0; j < 3; j++) {
                    XSSFCell cell = row.getCell(j);
                    switch (j) {
                        case 0: {
                            if (cell.getCellTypeEnum() != CellType.NUMERIC) {
                                if (!ObjectUtil.isEmpty(cell.getStringCellValue())) {
                                    return MyResult.failure("第" + i + "行,第" + 1 + "列数据有误");
                                } else {
                                    return MyResult.success(xnFoods);
                                }

                            } else {
                                xnFood.setWeek((int) cell.getNumericCellValue());
                            }
                        }
                        ;
                        break;
                        case 1: {
                            if (cell.getCellTypeEnum() != CellType.NUMERIC) {
                                return MyResult.failure("第" + i + "行,第" + 2 + "列数据有误");
                            } else {
                                xnFood.setFoodtime((byte) cell.getNumericCellValue());
                            }
                        }
                        ;
                        break;
                        case 2: {
                            if (cell.getCellTypeEnum() != CellType.STRING) {
                                return MyResult.failure("第" + i + "行,第" + 3 + "列数据有误");
                            } else {
                                xnFood.setFoodname(cell.getStringCellValue().trim());
                            }
                            xnFoods.add(xnFood);
                        }
                        ;
                        break;
                    }
                }

            }
            return MyResult.success(xnFoods);

        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }

}

并没有发现有什么地方有问题,于是我又去看看我的maven包有没有加入到项目里面

发现也没有问题。于是我又开始疯狂的百度。。有的说是版本不一致,但是我的都是一致的。。。。。。。

终于。整了我一天的时间。发现原来是这里没有导入包(Project Structure  --> artifacts --> 带exploded里面WEB-INF下lib没这个包)

加入那三个POI包之后,再次运行项目,OK。

--------------20191213更新

NoClassDefFoundError:可能出现的原因2:类里面包含了main方法。导致类无法初始化成功。删除main方法即可

 

<think> 我们遇到的问题是在使用POI库时出现的类型转换异常:`org.apache.poi.hssf.usermodel.HSSFSheet`不能转换为`org.apache.poi.xssf.streaming.SXSSFSheet`。这通常是由于同时操作两种不同格式的Excel文件(HSSF用于.xls格式,XSSF/SXSSF用于.xlsx格式)而导致的。 ### 问题原因分析 1. **混合使用不同格式的API** HSSF是POI中处理Excel 97-2003(.xls)格式的类,而SXSSF是处理Excel 2007+ (.xlsx)格式的流式API。两者不能互相转换[^1]。 2. **代码中未区分工作簿类型** 在操作工作表时,可能没有检查工作簿的实际类型,而直接将HSSFSheet当作SXSSFSheet使用。 3. **依赖冲突或版本问题** 如果项目中存在多个POI版本,或者依赖了不兼容的POI组件,也可能导致此异常[^2]。 ### 解决方案 #### 方案1:统一工作簿类型(推荐) 在创建或获取工作簿时,明确区分.xls和.xlsx格式: ```java // 读取文件时根据扩展名判断 Workbook workbook; if (fileName.endsWith(".xls")) { workbook = new HSSFWorkbook(); } else if (fileName.endsWith(".xlsx")) { // 使用SXSSFWorkbook处理大文件 workbook = new SXSSFWorkbook(); } else { throw new IllegalArgumentException("不支持的文件格式"); } // 获取工作表时同样需要类型判断 Sheet sheet = workbook.getSheetAt(0); if (workbook instanceof HSSFWorkbook) { HSSFSheet hssfSheet = (HSSFSheet) sheet; // 操作HSSFSheet } else if (workbook instanceof SXSSFWorkbook) { SXSSFSheet sxssfSheet = (SXSSFSheet) sheet; // 操作SXSSFSheet } ``` #### 方案2:使用通用接口 POI提供了统一的接口(如`Sheet`),应尽量使用通用接口方法: ```java // 声明为通用Sheet类型 Sheet sheet = workbook.getSheetAt(0); // 通用方法(两种格式都支持) int lastRowNum = sheet.getLastRowNum(); Row row = sheet.getRow(0); // 避免直接强制转换为具体实现类 ``` #### 方案3:检查依赖冲突 在Maven中检查POI依赖: ```xml <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>5.2.5</version> <!-- 统一版本 --> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>5.2.5</version> <!-- 保持相同版本 --> </dependency> ``` 运行`mvn dependency:tree`检查版本冲突,确保所有POI组件版本一致。 ### 错误代码示例分析 假设出现以下错误代码: ```java SXSSFSheet sheet = (SXSSFSheet) workbook.getSheetAt(0); // 当workbook是HSSFWorkbook时会崩溃 ``` 应改为: ```java Sheet sheet = workbook.getSheetAt(0); // 使用通用接口 if (sheet instanceof SXSSFSheet) { // SXSSFSheet特有操作 } else if (sheet instanceof HSSFSheet) { // HSSFSheet特有操作 } ``` ### 兼容性处理最佳实践 | 操作类型 | 通用API | 特有API处理方式 | |----------------|--------------------------|------------------------------------------| | 创建工作簿 | `WorkbookFactory.create` | 根据文件扩展名选择实现类 | | 获取单元格值 | `Cell.getCellType()` | 使用`CellType`枚举而非旧版`int`常量 | | 设置单元格样式 | `Cell.setCellStyle()` | 样式对象需通过对应工作簿创建 | > 📌 **关键原则**:始终通过工作簿实例创建依赖对象(如样式、字体等),避免跨工作簿类型使用对象[^3]。 §§相关问题§§ 1. 如何通过POI识别Excel文件格式(.xls/.xlsx)? 2. 为什么POI在处理Excel时会占用大量内存? 3. POI中HSSF和XSSF的主要性能差异是什么? 4. 如何解决POI版本冲突导致的NoSuchMethodError? [^1]: POI官方文档明确指出HSSF和XSSF/SXSSF互不兼容 [^2]: 常见于同时使用EasyExcel和POI直接操作的情况 [^3]: 样式对象绑定到特定工作簿实例 </think> ### 解决`HSSFSheet`无法转换为`SXSSFSheet`的ClassCastException问题 #### 📌 问题根本原因 1. **格式混用冲突** `HSSFSheet`是处理`.xls`格式(Excel 97-2003)的API,而`SXSSFSheet`是处理`.xlsx`格式(Excel 2007+)的流式API。两者属于不同的实现体系,不能直接转换[^1]。 2. **代码未区分工作簿类型** 在操作工作表时,未检查工作簿的实际类型(HSSFWorkbook 或 SXSSFWorkbook),直接进行强制类型转换。 3. **依赖版本冲突** 项目中可能存在多个POI版本,低版本API在高版本环境中调用时会出现兼容性问题[^2]。 #### 🔧 解决方案 ##### 方案1:使用通用接口(推荐) ```java Workbook workbook = ... // 获取工作簿实例 // ✅ 安全方式:通过接口操作 Sheet sheet = workbook.getSheetAt(0); // 使用通用Sheet接口 // 避免直接强制转换 // ❌ SXSSFSheet sxSheet = (SXSSFSheet) workbook.getSheetAt(0); ``` ##### 方案2:类型检查后再转换 ```java if (workbook instanceof SXSSFWorkbook) { SXSSFSheet sxSheet = ((SXSSFWorkbook) workbook).getSheetAt(0); // 执行.xlsx操作 } else if (workbook instanceof HSSFWorkbook) { HSSFSheet hssfSheet = ((HSSFWorkbook) workbook).getSheetAt(0); // 执行.xls操作 } ``` ##### 方案3:统一工作簿创建方式 创建时明确指定格式: ```java // 创建.xlsx工作簿(使用SXSSF) Workbook workbook = new SXSSFWorkbook(); // 创建.xls工作簿(使用HSSF) Workbook workbook = new HSSFWorkbook(); ``` #### ⚠️ 依赖冲突处理 在`pom.xml`中统一POI版本: ```xml <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>5.2.5</version> <!-- 统一版本号 --> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>5.2.5</version> <!-- 必须相同 --> </dependency> ``` 执行依赖检查: ```bash mvn dependency:tree | grep poi ``` #### 💡 最佳实践 1. **文件扩展名检查** ```java public Workbook createWorkbook(String filePath) { if (filePath.endsWith(".xlsx")) { return new SXSSFWorkbook(); // 流式处理大文件 } else if (filePath.endsWith(".xls")) { return new HSSFWorkbook(); } throw new IllegalArgumentException("Unsupported file format"); } ``` 2. **使用工厂模式** ```java Workbook workbook = WorkbookFactory.create(new File("data.xlsx")); ``` 3. **避免跨类型操作** ```java // 错误示例:混合创建 Workbook wb = new HSSFWorkbook(); Sheet sheet = wb.createSheet("Data"); SXSSFSheet streamingSheet = (SXSSFSheet) sheet; // 抛出ClassCastException ``` #### 📊 技术对比 | 特性 | HSSF (.xls) | SXSSF (.xlsx) | |---------------------|-------------------|---------------------| | 最大行数 | 65,536 | 1,048,576 | | 内存占用 | 高 | 低(流式处理) | | 适用场景 | 小文件 | 大数据量导出 | | 实现类 | HSSFSheet | SXSSFSheet | > **关键提示**:POI中所有工作簿实现都实现`Workbook`接口,所有工作表实现`Sheet`接口。应优先使用接口方法避免类型转换[^3]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值