【发布时间】:2015-02-17 14:46:00
【问题描述】:
请在标记为重复之前阅读全文。我已经看过所有相关/类似的问题,我想我这里发生了一些奇怪的事情。
我正在使用 Apache POI 3.9 版来生成 4 张工作簿。每次我填完表格时,我都会自动调整列的大小。代码如下:
for (String alias : map.keySet()) {
Sheet sheet = workbook.createSheet(getSheetNameForAlias(alias));
List<Object[]> resultList = (List<Object[]>) map.get(alias);
Collections.sort(resultList.subList(1, resultList.size()), getComparator(alias));
int rownum = 0;
for (Object[] array : resultList) {
rownum = populateRow(cellStyles, sheet, rownum, array, false);
}
for(int j=0; j< resultList.get(0).length; j++){
sheet.autoSizeColumn(j, false);
}
}
populateRow 函数是:
private int populateRow(CellStyle[] cellStyles, Sheet sheet, int rownum, Object[] dataArr, boolean doAutoSizing) {
if (logger.isDebugEnabled()) {
logger.debug("Populating row ... @{} : {} ", sheet.getSheetName(), rownum);
}
int cellnum;
Row row = sheet.createRow(rownum++);
cellnum = 0;
if (ArrayUtils.isEmpty(dataArr)) {
logger.error("No data found for row ... @{} : {} ", sheet.getSheetName(), rownum);
return rownum;
}
for (Object obj : dataArr) {
Cell cell = row.createCell(cellnum++);
// get the type of the data inserted in the cell.
// Accordingly set the cell value
String data = String.valueOf(obj);
if (StringUtils.isBlank(data) || "null".equals(data)) {
cell.setCellValue(StringUtils.EMPTY);
} else {
if (NumberUtils.isNumber(data)) {
Double dbVal = NumberUtils.createDouble(data);
// Millions separator for Numeric fields
if (dbVal == Math.ceil(dbVal)) {
cell.setCellStyle(cellStyles[0]);
} else {
cell.setCellStyle(cellStyles[1]);
}
cell.setCellType(Cell.CELL_TYPE_NUMERIC); // for numeric , set cell type as numeric
cell.setCellValue(dbVal);
} else {
if(obj instanceof java.sql.Timestamp || obj instanceof java.sql.Date){
cell.setCellStyle(cellStyles[2]);
}
cell.setCellValue(data);
}
if (doAutoSizing) {
sheet.autoSizeColumn(cellnum - 1);
}
data = null;
}
obj = null;
}
return rownum;
}
奇怪的是,四张纸中的一张的列大小调整不正确。
我尝试在脑海中用不同的方法重写auto size 逻辑,并发现如果我尝试sheet.getRow(0),它会给出null 指定该行没有数据,但生成的 excel 并不适用,所有四张表都有适当的数据。在每种情况下,第一行始终是标题,因此即使其下方没有数据,它也不能为空。
我不知道我错过了什么,但我尝试并发现这不是 JVM 的字体问题。
如果您需要其他信息,请告诉我。
编辑: 发现自动调整大小对于超过 100 行的工作表无法正常工作。我不知道如何将其与某些逻辑联系起来。这真是太奇怪了。
编辑 2:我在寻找其他东西时找到了 Axel 给出的原因的官方参考。这确实足以解决我的困惑。阅读以下段落 link
【问题讨论】:
-
很难说,假设您的
dataArr每行包含相同数量的元素,我觉得还可以。也许添加一些日志记录以指示每次迭代的数组大小。问题:总是同一张纸吗?哪个(第一个,最后一个......)?是否有更多相关的代码,例如您可能正在清除该行? -
@geert3 请查看编辑。它发生在超过 100 行的工作表上,其他工作正常。是的,
dataArr每行包含相同数量的元素。 -
在 poi 3.10 和 3.7 中对我来说效果很好 - 如果您可以发布更完整的源代码,也许还有一些输入数据。您还可以将 println 添加到围绕 populateRow 和 autoSizeColumn 的两个主要
for循环中并显示该输出。 -
请显示创建工作簿的代码。我有一个想法,需要知道您的案例中使用的构造函数/工厂。
-
感谢大家的及时回复。明天会添加详细信息,因为我是 OOO。
标签: java excel apache-poi