gpt4 book ai didi

java - Apache POI - FileInputStream 工作,文件对象失败(NullPointerException)

转载 作者:搜寻专家 更新时间:2023-11-01 01:20:20 28 4
gpt4 key购买 nike

我尝试将所有工作表从一个工作簿复制到另一个工作簿。问题是,如果我通过 FileInputStreams 读取工作簿,它可以正常工作,但它不适用于文件对象。

考虑以下方法:

import java.io.BufferedReader;
import java.io.File;
import java.io.FileFilter;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;
import java.util.ArrayList;
import java.util.Iterator;

import org.apache.commons.io.IOUtils;
import org.apache.commons.io.filefilter.WildcardFileFilter;
import org.apache.poi.EncryptedDocumentException;
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
import org.apache.poi.ss.SpreadsheetVersion;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.CellType;
import org.apache.poi.ss.usermodel.DataConsolidateFunction;
import org.apache.poi.ss.usermodel.DateUtil;
import org.apache.poi.ss.usermodel.Font;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.ss.usermodel.WorkbookFactory;
import org.apache.poi.ss.util.AreaReference;
import org.apache.poi.ss.util.CellReference;
import org.apache.poi.xssf.usermodel.XSSFPivotTable;
import org.apache.poi.xssf.usermodel.XSSFSheet;


public void copyAllSheetsAcrossWorkbook(String oldWorkbook, String newWorkbook)
throws EncryptedDocumentException, InvalidFormatException, IOException {
FileInputStream fisOld = null;
FileInputStream fisNew = null;
Workbook oldWB = null;
Workbook newWB = null;
FileOutputStream fileOut = null;

System.out.println("oldWorkbook: " + oldWorkbook);
System.out.println("newWorkbook: " + newWorkbook);
fisOld = new FileInputStream(oldWorkbook);
fisNew = new FileInputStream(newWorkbook);

// THIS WORKS
// oldWB = WorkbookFactory.create(fisOld);
// newWB = WorkbookFactory.create(fisNew);

// THIS DOES NOT WORK
oldWB = WorkbookFactory.create(new File(oldWorkbook));
newWB = WorkbookFactory.create(new File(newWorkbook));

if (newWB == null) {
System.out.println("newWB is null");
}
// CellStyle newStyle = newWB.createCellStyle();
Row row;
Cell cell;
copiedSheets = new ArrayList<String>();
for (int i = 0; i < oldWB.getNumberOfSheets(); i++) {
XSSFSheet sheetFromOldWB = (XSSFSheet) oldWB.getSheetAt(i);
String sheetNameFromOldWB = sheetFromOldWB.getSheetName();
XSSFSheet sheetForNewWB = (XSSFSheet) newWB.getSheet(sheetNameFromOldWB);
if (sheetForNewWB != null) {
int sheetIndex = newWB.getSheetIndex(sheetNameFromOldWB);
newWB.removeSheetAt(sheetIndex);
}
LOGGER.info("Copying to new Workbook: " + sheetNameFromOldWB);
sheetForNewWB = (XSSFSheet) newWB.createSheet(sheetFromOldWB.getSheetName());
for (int rowIndex = 0; rowIndex < sheetFromOldWB.getPhysicalNumberOfRows(); rowIndex++) {
row = sheetForNewWB.createRow(rowIndex);
for (int colIndex = 0; colIndex < sheetFromOldWB.getRow(rowIndex).getPhysicalNumberOfCells(); colIndex++) {
cell = row.createCell(colIndex);
// get cell from old WB's sheet and when cell is null, return as blank cells.
Cell c = sheetFromOldWB.getRow(rowIndex).getCell(colIndex, Row.MissingCellPolicy.CREATE_NULL_AS_BLANK);

// Below is where all the copying is happening.
// CellStyle origStyle = c.getCellStyle();
// newStyle.cloneStyleFrom(origStyle);
// cell.setCellStyle(newStyle);
switch (c.getCellTypeEnum()) {
case STRING:
cell.setCellValue(c.getRichStringCellValue().getString());
break;
case NUMERIC:
if (DateUtil.isCellDateFormatted(cell)) {
cell.setCellValue(c.getDateCellValue());
} else {
cell.setCellValue(c.getNumericCellValue());
}
break;
case BOOLEAN:
cell.setCellValue(c.getBooleanCellValue());
break;
case FORMULA:
cell.setCellFormula(c.getCellFormula());
break;
default:
break;
}
}
}
copiedSheets.add(oldWB.getSheetName(i));

}
fileOut = new FileOutputStream(newWorkbook);
newWB.write(fileOut); // <------ HERE I GET NULLPOINTEREXCEPTION
fisOld.close();
fisNew.close();
oldWB.close();
fileOut.close();
newWB.close();

我在 newWB.write(fileOut); 处得到以下异常:

Exception in thread "main" org.apache.poi.POIXMLException: java.lang.NullPointerException
at org.apache.poi.POIXMLDocument.getProperties(POIXMLDocument.java:168)
at org.apache.poi.POIXMLDocument.write(POIXMLDocument.java:246)
at com.capgemini.toolkit.App.copyAllSheetsAcrossWorkbook(App.java:263)
at com.capgemini.toolkit.App.main(App.java:58)

Caused by: java.lang.NullPointerException
at org.apache.poi.openxml4j.util.ZipSecureFile$ThresholdInputStream.read(ZipSecureFile.java:210)
at com.sun.org.apache.xerces.internal.impl.XMLEntityManager$RewindableInputStream.read(Unknown Source)
at com.sun.org.apache.xerces.internal.impl.XMLEntityManager.setupCurrentEntity(Unknown Source)
at com.sun.org.apache.xerces.internal.impl.XMLVersionDetector.determineDocVersion(Unknown Source)
at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(Unknown Source)
at com.sun.org.apache.xerces.internal.parsers.XML11Configuration.parse(Unknown Source)
at com.sun.org.apache.xerces.internal.parsers.XMLParser.parse(Unknown Source)
at com.sun.org.apache.xerces.internal.parsers.DOMParser.parse(Unknown Source)
at com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderImpl.parse(Unknown Source)
at javax.xml.parsers.DocumentBuilder.parse(Unknown Source)
at org.apache.poi.util.DocumentHelper.readDocument(DocumentHelper.java:140)
at org.apache.poi.POIXMLTypeLoader.parse(POIXMLTypeLoader.java:143)
at org.openxmlformats.schemas.officeDocument.x2006.extendedProperties.PropertiesDocument$Factory.parse(Unknown Source)
at org.apache.poi.POIXMLProperties.<init>(POIXMLProperties.java:78)
at org.apache.poi.POIXMLDocument.getProperties(POIXMLDocument.java:166)
... 3 more

在 POI 文档中,总是提到更好地使用 File 对象,因为它可以降低内存消耗。这就是为什么我想知道为什么它不适用于 File 对象。

为了测试,这是在主要方法中运行的唯一方法,我使用了 2 个带有一些虚拟数据的新 Excel 文件 (.xlsx)。

有人知道为什么它不能与 File 对象一起使用吗?我做错了什么吗?

仅供引用:我正在使用 POI 3.16。

最佳答案

使用 File 而不是 FileInputStream 打开 Workbook 可以减少内存占用,因为如果是 XSSF(*.xlsx), ZipPackage将直接从 *.xlsx 文件打开,而不是将整个 ZIP 内容读入内存。

但这也意味着,ZipPackage 会打开文件,直到 Workbook 关闭。因此,在 Workbook 关闭之前,没有任何内容可以同时写入该文件。因此,由于无法使用 FileWorkbook 内容写回到打开 Workbook 的同一文件中相反,如果您只想从该 Workbook 中读取,则用于打开 WorkbookFileInputStream 就可以了。但如果你想读取和写入同一个文件,它就不起作用了。然后需要 FileInputStreamFileOutputStream

因此,在您的情况下,您尝试从 File 读取 Workbook newWB,然后使用 WorkbookWorkbook 写入同一文件

fileOut = new FileOutputStream(newWorkbook);
newWB.write(fileOut);

文件已经打开。这失败了。

但是:

   fisNew = new FileInputStream(newWorkbook);
oldWB = WorkbookFactory.create(new File(oldWorkbook));
newWB = WorkbookFactory.create(fisNew);
...
fileOut = new FileOutputStream(newWorkbook);
newWB.write(fileOut);
fileOut.close();

oldWB.close();
newWB.close();

应该可以。

顺便说一句:如果您使用的是 File,那么您不应该对同一个文件使用 FileInputStream。所以不要使用 fisOld

使用 File 而不是 FileInputStream 打开 Workbook 的另一个缺点是,在关闭 Workbook 因此隐式关闭底层文件系统(POIFSFileSystemHSSF 的情况下和 ZipPackageXSSF 的情况下)该文件获得更新的最后修改日期。没有对文件进行任何更改,但文件已被打开并新写入文件系统。这就是更新最后修改日期的原因。


2017 年 9 月 21 日编辑:使用 File 的缺点似乎比最初想象的要严重。 OPCPackage.close还将所有更改保存到底层 OPCPackage 中。因此,如果您从一个文件打开一个 XSSFWorkbook,然后想使用 write(java.io.OutputStream stream) 将更改写入另一个文件,那么源文件也将在关闭 OPCPackage 时更改。仅当从 XSSFWorkbook 使用 write(java.io.OutputStream stream) 时才会出现此问题 POIXMLDocument.write被称为POIXMLDocumentPart.onSave其中“保存底层 OOXML 包中的更改。”。因此 OPCPackage 在关闭前更新了所有更改。

简短示例:

import org.apache.poi.ss.usermodel.*;

import java.io.File;
import java.io.FileOutputStream;

class ReadAndWriteExcelWorkbook {

public static void main(String[] args) throws Exception {

Workbook workbook = WorkbookFactory.create(new File("file.xlsx"));

Sheet sheet = workbook.getSheetAt(0);
Row row = sheet.getRow(0);
if (row == null) row = sheet.createRow(0);
Cell cell = row.getCell(0);
if (cell == null) cell = row.createCell(0);
cell.setCellValue("changed");

FileOutputStream out = new FileOutputStream("fileNew.xlsx");
workbook.write(out);
out.close();
workbook.close();

}
}

在这段代码之后,fileNew.xlsxfile.xlsx 都发生了变化。

关于java - Apache POI - FileInputStream 工作,文件对象失败(NullPointerException),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46146161/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com