java.lang.NoSuchFieldError: Fact
2024-07-16 本文已影响0人
奔腾的小溪
背景
需要使用MultipartFile来接收doc、docx类型的文件,并对文件的内容进行校验是否包含某些模板的文字。
报错过程
1、文件类型不适配
- 接收文件时docx可以上传成功,但是doc类型报错①如下:
Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is org.apache.poi.openxml4j.exceptions.OLE2NotOfficeXmlFileException: The supplied data appears to be in the OLE2 Format. You are calling the part of POI that deals with OOXML (Office Open XML) Documents. You need to call a different part of POI to process this data (eg HSSF instead of XSSF)] with root cause org.apache.poi.openxml4j.exceptions.OLE2NotOfficeXmlFileException: The supplied data appears to be in the OLE2 Format. You are calling the part of POI that deals with OOXML (Office Open XML) Documents. You need to call a different part of POI to process this data (eg HSSF instead of XSSF) - 报错解决方案:
使用Apache POI的HWPFDocument 来读取doc文件,使用XWPFDocument 读取docx文件。
try (InputStream is = multipartFile.getInputStream()) {
// 将MultipartFile转换为HWPFDocument
HWPFDocument doc = new HWPFDocument(is);
Range textRange = doc.getRange();
String text = textRange.text();
}
`
try (InputStream is = multipartFile.getInputStream()) {
XWPFDocument document = new XWPFDocument(multipartFile.getInputStream());
XWPFWordExtractor extractor = new XWPFWordExtractor(document);
String text = extractor.getText();
}
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-scratchpad</artifactId>
<version>5.2.3</version>
</dependency>
<!-- Apache POI for DOCX files -->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>5.2.3</version>
</dependency>
2、pom引用问题
- 报错②
java.lang.NoSuchFieldError: Factory
at org.apache.poi.xddf.usermodel.chart.XDDFChart.<init>(XDDFChart.java:152) ~[poi-ooxml-5.2.3.jar:5.2.3]
- 解决方案
Ⅰ 尝试提高pom版本
<version>5.2.4</version>
结果引入了新的报错③
java.lang.NoSuchMethodError: org.apache.commons.io.output.UnsynchronizedByteArrayOutputStream.builder()Lorg/apache/commons/io/output/UnsynchronizedByteArrayOutputStream$Builder
查了下因为commons.io 版本过低导致的。(https://blog.csdn.net/weixin_48524970/article/details/134379243)
修改commons.io版本为2.13.0后报错③不见了,报错②还在,说明单纯的升级版本解决不了问题。
Ⅱ 终极原因:jar包冲突导致
回退pom版本及commons.io版本。
(https://blog.csdn.net/Wolf__king/article/details/132078098)
1、先百度java.lang.NoSuchFieldError这个异常的意思,得知这个跟jar包冲突有关系
2、根据上述的截图,知道冲突的方法所在的jar包是poi-ooml,找到这一行代码,发现是存在这个factory的。
3、怀疑服务上有两个版本的这个jar包导致的,使用的版本不是预期的这个版本找不到这个代码。登录服务器确认,这个jar包只有这一个版本,怀疑点不对。
4、在从代码入手,查看factory所在的类,ThemeDocument
全代码搜索这个类,Idea的快捷键(ctrl+N)发现有两个jar包包含这个类,poi-ooxml-lite和前面提到的poi-ooxml-schemas这个jar包。
5、修改poi-ooxml-schemas版本为 4.1.2 问题解决
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml-schemas</artifactId>
<version>4.1.2</version>
</dependency>
6、最终的pom为
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-scratchpad</artifactId>
<version>5.2.3</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>5.2.3</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>5.2.3</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml-schemas</artifactId>
<version>4.1.2</version>
</dependency>