cn.hutool.poi.excel.ExcelWriter Maven / Gradle / Ivy
package cn.hutool.poi.excel;
import java.io.Closeable;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.TreeMap;
import java.util.concurrent.atomic.AtomicInteger;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.io.FileUtil;
import cn.hutool.core.io.IORuntimeException;
import cn.hutool.core.io.IoUtil;
import cn.hutool.core.lang.Assert;
import cn.hutool.core.map.MapUtil;
/**
* Excel 写入器
* 此工具用于通过POI将数据写出到Excel
*
* @author Looly
* @since 3.2.0
*/
public class ExcelWriter implements Closeable {
/** 是否被关闭 */
private boolean isClosed;
/** 工作簿 */
private Workbook workbook;
/** Excel中对应的Sheet */
private Sheet sheet;
/** 目标文件 */
private File destFile;
/** 当前行 */
private AtomicInteger currentRow = new AtomicInteger(0);
/** 标题行别名 */
private Map headerAlias;
/** 样式集,定义不同类型数据样式 */
private StyleSet styleSet;
// -------------------------------------------------------------------------- Constructor start
/**
* 构造,默认生成xls格式的Excel文件
* 此构造不传入写出的Excel文件路径,只能调用{@link #flush(OutputStream)}方法写出到流
* 若写出到文件,还需调用{@link #setDestFile(File)}方法自定义写出的文件,然后调用{@link #flush()}方法写出到文件
*
* @since 3.2.1
*/
public ExcelWriter() {
this(false);
}
/**
* 构造
* 此构造不传入写出的Excel文件路径,只能调用{@link #flush(OutputStream)}方法写出到流
* 若写出到文件,还需调用{@link #setDestFile(File)}方法自定义写出的文件,然后调用{@link #flush()}方法写出到文件
*
* @param isXlsx 是否为xlsx格式
* @since 3.2.1
*/
public ExcelWriter(boolean isXlsx) {
this(WorkbookUtil.createBook(isXlsx ? ".xlsx" : ".xls"), null);
}
/**
* 构造,默认写出到第一个sheet,第一个sheet名为sheet1
*
* @param destFilePath 目标文件路径,可以不存在
*/
public ExcelWriter(String destFilePath) {
this(destFilePath, null);
}
/**
* 构造
*
* @param destFilePath 目标文件路径,可以不存在
* @param sheetName sheet名,第一个sheet名并写出到此sheet,例如sheet1
*/
public ExcelWriter(String destFilePath, String sheetName) {
this(FileUtil.file(destFilePath), sheetName);
}
/**
* 构造,默认写出到第一个sheet,第一个sheet名为sheet1
*
* @param destFile 目标文件,可以不存在
*/
public ExcelWriter(File destFile) {
this(destFile, null);
}
/**
* 构造
*
* @param destFile 目标文件,可以不存在
* @param sheetName sheet名,做为第一个sheet名并写出到此sheet,例如sheet1
*/
public ExcelWriter(File destFile, String sheetName) {
this(WorkbookUtil.createBook(destFile), sheetName);
this.destFile = destFile;
}
/**
* 构造
* 此构造不传入写出的Excel文件路径,只能调用{@link #flush(OutputStream)}方法写出到流
* 若写出到文件,还需调用{@link #setDestFile(File)}方法自定义写出的文件,然后调用{@link #flush()}方法写出到文件
*
* @param workbook {@link Workbook}
* @param sheetName sheet名,做为第一个sheet名并写出到此sheet,例如sheet1
*/
public ExcelWriter(Workbook workbook, String sheetName) {
this(ExcelUtil.getOrCreateSheet(workbook, sheetName));
}
/**
* 构造
* 此构造不传入写出的Excel文件路径,只能调用{@link #flush(OutputStream)}方法写出到流
* 若写出到文件,还需调用{@link #setDestFile(File)}方法自定义写出的文件,然后调用{@link #flush()}方法写出到文件
*
* @param sheet {@link Sheet}
* @since 4.0.6
*/
public ExcelWriter(Sheet sheet) {
this.workbook = sheet.getWorkbook();
this.sheet = sheet;
this.styleSet = new StyleSet(workbook);
}
// -------------------------------------------------------------------------- Constructor end
/**
* 获取Workbook
*
* @return Workbook
*/
public Workbook getWorkbook() {
return this.workbook;
}
/**
* 获取当前Sheet
*
* @return {@link Sheet}
*/
public Sheet getSheet() {
return this.sheet;
}
/**
* 设置某列为自动宽度
*
* @param columnIndex 第几列,从0计数
* @param useMergedCells 是否适用于合并单元格
* @return this
* @since 3.3.0
*/
public ExcelWriter autoSizeColumn(int columnIndex, boolean useMergedCells) {
this.sheet.autoSizeColumn(columnIndex, useMergedCells);
return this;
}
/**
* 获取样式集,样式集可以自定义包括:
*
* 1. 头部样式
* 2. 一般单元格样式
* 3. 默认数字样式
* 4. 默认日期样式
*
*
* @return 样式集
* @since 4.0.0
*/
public StyleSet getStyleSet() {
return this.styleSet;
}
/**
* 获取头部样式,获取样式后可自定义样式
*
* @return 头部样式
*/
public CellStyle getHeadCellStyle() {
return this.styleSet.headCellStyle;
}
/**
* 获取单元格样式,获取样式后可自定义样式
*
* @return 单元格样式
*/
public CellStyle getCellStyle() {
return this.styleSet.cellStyle;
}
/**
* 获得当前行
*
* @return 当前行
*/
public int getCurrentRow() {
return this.currentRow.get();
}
/**
* 设置当前所在行
*
* @param rowIndex 行号
* @return this
*/
public ExcelWriter setCurrentRow(int rowIndex) {
this.currentRow.set(rowIndex);
return this;
}
/**
* 跳过当前行
*
* @return this
*/
public ExcelWriter passCurrentRow() {
this.currentRow.incrementAndGet();
return this;
}
/**
* 跳过指定行数
*
* @param rows 跳过的行数
* @return this
*/
public ExcelWriter passRows(int rows) {
this.currentRow.addAndGet(rows);
return this;
}
/**
* 重置当前行为0
*
* @return this
*/
public ExcelWriter resetRow() {
this.currentRow.set(0);
return this;
}
/**
* 切换sheet,如果指定的sheet不存在,创建之
* @param sheetName sheet名
* @return this
* @since 4.0.8
*/
public ExcelWriter setOrCreateSheet(String sheetName) {
this.sheet = ExcelUtil.getOrCreateSheet(this.workbook, sheetName);
return this;
}
/**
* 设置写出的目标文件
*
* @param destFile 目标文件
* @return this
*/
public ExcelWriter setDestFile(File destFile) {
this.destFile = destFile;
return this;
}
/**
* 设置标题别名,key为Map中的key,value为别名
*
* @param headerAlias 标题别名
* @return this
* @since 3.2.1
*/
public ExcelWriter setHeaderAlias(Map headerAlias) {
this.headerAlias = headerAlias;
return this;
}
/**
* 设置列宽(单位为一个字符的宽度,例如传入width为10,表示10个字符的宽度)
*
* @param columnIndex 列号(从0开始计数,-1表示所有列的默认宽度)
* @param width 宽度(单位1~256个字符宽度)
* @return this
* @since 4.0.8
*/
public ExcelWriter setColumnWidth(int columnIndex, int width) {
if(columnIndex < 0) {
this.sheet.setDefaultColumnWidth(width);
}else {
this.sheet.setColumnWidth(columnIndex, width * 256);
}
return this;
}
/**
* 设置行高,值为一个点的高度
*
* @param rownum 行号(从0开始计数,-1表示所有行的默认高度)
* @param height 高度
* @return this
* @since 4.0.8
*/
public ExcelWriter setRowHeight(int rownum, int height) {
if(rownum < 0) {
this.sheet.setDefaultRowHeightInPoints(height);
}else {
this.sheet.getRow(rownum).setHeightInPoints(height);
}
return this;
}
/**
* 合并当前行的单元格
* 样式为默认标题样式,可使用{@link #getHeadCellStyle()}方法调用后自定义默认样式
*
* @param lastColumn 合并到的最后一个列号
* @return this
*/
public ExcelWriter merge(int lastColumn) {
return merge(lastColumn, null);
}
/**
* 合并当前行的单元格,并写入对象到单元格
* 如果写到单元格中的内容非null,行号自动+1,否则当前行号不变
* 样式为默认标题样式,可使用{@link #getHeadCellStyle()}方法调用后自定义默认样式
*
* @param lastColumn 合并到的最后一个列号
* @param content 合并单元格后的内容
* @return this
*/
public ExcelWriter merge(int lastColumn, Object content) {
return merge(lastColumn, content, true);
}
/**
* 合并某行的单元格,并写入对象到单元格
* 如果写到单元格中的内容非null,行号自动+1,否则当前行号不变
* 样式为默认标题样式,可使用{@link #getHeadCellStyle()}方法调用后自定义默认样式
*
* @param lastColumn 合并到的最后一个列号
* @param content 合并单元格后的内容
* @param isSetHeaderStyle 是否为合并后的单元格设置默认标题样式
* @return this
* @since 4.0.10
*/
public ExcelWriter merge(int lastColumn, Object content, boolean isSetHeaderStyle) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
final int rowIndex = this.currentRow.get();
merge(rowIndex, rowIndex, 0, lastColumn, content, isSetHeaderStyle);
// 设置内容后跳到下一行
if(null != content) {
this.currentRow.incrementAndGet();
}
return this;
}
/**
* 合并某行的单元格,并写入对象到单元格
* 如果写到单元格中的内容非null,行号自动+1,否则当前行号不变
* 样式为默认标题样式,可使用{@link #getHeadCellStyle()}方法调用后自定义默认样式
*
* @param lastColumn 合并到的最后一个列号
* @param content 合并单元格后的内容
* @param isSetHeaderStyle 是否为合并后的单元格设置默认标题样式
* @return this
* @since 4.0.10
*/
public ExcelWriter merge(int firstRow, int lastRow, int firstColumn, int lastColumn , Object content, boolean isSetHeaderStyle) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
final CellStyle style = (isSetHeaderStyle && null != this.styleSet.headCellStyle) ? this.styleSet.headCellStyle : this.styleSet.cellStyle;
CellUtil.mergingCells(this.sheet, firstRow, lastRow, firstColumn, lastColumn, style);
// 设置内容
if (null != content) {
final Cell cell = getOrCreateCell(firstColumn, firstRow);
CellUtil.setCellValue(cell, content, this.styleSet);
if(isSetHeaderStyle) {
//当作为标题合并单元格时,使用标题样式
cell.setCellStyle(this.styleSet.headCellStyle);
}
}
return this;
}
/**
* 写出数据,本方法只是将数据写入Workbook中的Sheet,并不写出到文件
* 写出的起始行为当前行号,可使用{@link #getCurrentRow()}方法调用,根据写出的的行数,当前行号自动增加
* 样式为默认样式,可使用{@link #getCellStyle()}方法调用后自定义默认样式
* data中元素支持的类型有:
*
*
* 1. Iterable,既元素为一个集合,元素被当作一行,data表示多行
* 2. Map,既元素为一个Map,第一个Map的keys作为首行,剩下的行为Map的values,data表示多行
* 3. Bean,既元素为一个Bean,第一个Bean的字段名列表会作为首行,剩下的行为Bean的字段值列表,data表示多行
* 4. 无法识别,不输出
*
*
* @param data 数据
* @return this
*/
public ExcelWriter write(Iterable> data) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
int index = 0;
for (Object object : data) {
if (object instanceof Iterable) {
// 普通多行数据
writeRow((Iterable>) object);
} else if (object instanceof Map) {
// Map表示一行,第一条数据的key做为标题行
writeRows((Map, ?>) object, 0 == index);
} else if (BeanUtil.isBean(object.getClass())) {
// 一个Bean对象表示一行
writeRows(BeanUtil.beanToMap(object), 0 == index);
} else {
break;
}
index++;
}
if (0 == index) {
// 在无法识别元素类型的情况下,做为一行对待
writeRow(data);
}
return this;
}
/**
* 写出数据,本方法只是将数据写入Workbook中的Sheet,并不写出到文件
* 写出的起始行为当前行号,可使用{@link #getCurrentRow()}方法调用,根据写出的的行数,当前行号自动增加
* 样式为默认样式,可使用{@link #getCellStyle()}方法调用后自定义默认样式
* data中元素支持的类型有:
*
*
* 1. Map,既元素为一个Map,第一个Map的keys作为首行,剩下的行为Map的values,data表示多行
* 2. Bean,既元素为一个Bean,第一个Bean的字段名列表会作为首行,剩下的行为Bean的字段值列表,data表示多行
*
*
* @param data 数据
* @param comparator 比较器
* @return this
* @since 3.2.3
*/
@SuppressWarnings({ "rawtypes", "unchecked" })
public ExcelWriter write(Iterable data, Comparator comparator) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
boolean isFirstRow = true;
Map, ?> map;
for (T t : data) {
if (t instanceof Map) {
map = new TreeMap<>(comparator);
map.putAll((Map) t);
} else {
map = BeanUtil.beanToMap(t, new TreeMap(comparator), false, false);
}
writeRows(map, isFirstRow);
if (isFirstRow) {
isFirstRow = false;
}
}
return this;
}
/**
* 写出一行标题数据
* 本方法只是将数据写入Workbook中的Sheet,并不写出到文件
* 写出的起始行为当前行号,可使用{@link #getCurrentRow()}方法调用,根据写出的的行数,当前行号自动+1
* 样式为默认标题样式,可使用{@link #getHeadCellStyle()}方法调用后自定义默认样式
*
* @param rowData 一行的数据
* @return this
*/
public ExcelWriter writeHeadRow(Iterable> rowData) {
RowUtil.writeRow(this.sheet.createRow(this.currentRow.getAndIncrement()), rowData, this.styleSet);
return this;
}
/**
* 写出一行数据
* 本方法只是将数据写入Workbook中的Sheet,并不写出到文件
* 写出的起始行为当前行号,可使用{@link #getCurrentRow()}方法调用,根据写出的的行数,当前行号自动+1
* 样式为默认样式,可使用{@link #getCellStyle()}方法调用后自定义默认样式
*
* @param rowData 一行的数据
* @return this
*/
public ExcelWriter writeRow(Iterable> rowData) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
RowUtil.writeRow(this.sheet.createRow(this.currentRow.getAndIncrement()), rowData, this.styleSet);
return this;
}
/**
* 将一个Map写入到Excel,isWriteKeys为true写出两行,Map的keys做为一行,values做为第二行,否则只写出一行values
*
* @param rowMap 写出的Map
* @param isWriteKeys 为true写出两行,Map的keys做为一行,values做为第二行,否则只写出一行values
* @return this
*/
public ExcelWriter writeRows(Map, ?> rowMap, boolean isWriteKeys) {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
if (isWriteKeys) {
writeHeadRow(aliasHeader(rowMap.keySet()));
}
writeRow(rowMap.values());
return this;
}
/**
* 给指定单元格赋值
* @param x X坐标,从0计数,既列号
* @param y Y坐标,从0计数,既行号
* @param value 值
* @return this
* @since 4.0.2
*/
public ExcelWriter writeCellValue(int x, int y, Object value) {
final Cell cell = getOrCreateCell(x, y);
CellUtil.setCellValue(cell, value, styleSet);
return this;
}
/**
* 获取或者创建指定位置的单元格
*
* @param x X坐标,从0计数,既列号
* @param y Y坐标,从0计数,既行号
* @return {@link Cell}
* @since 4.0.9
*/
public Cell getOrCreateCell(int x, int y) {
final Row row = RowUtil.getOrCreateRow(this.sheet, y);
return CellUtil.getOrCreateCell(row, x);
}
/**
*
* @param x X坐标,从0计数,既列号
* @param y Y坐标,从0计数,既行号
* @return {@link CellStyle}
* @since 4.0.9
*/
public CellStyle createStyleForCell(int x, int y) {
final CellStyle cellStyle = this.workbook.createCellStyle();
final Cell cell = getOrCreateCell(x, y);
cell.setCellStyle(this.workbook.createCellStyle());
return cellStyle;
}
/**
* 将Excel Workbook刷出到预定义的文件
* 如果用户未自定义输出的文件,将抛出{@link NullPointerException}
* 预定义文件可以通过{@link #setDestFile(File)} 方法预定义,或者通过构造定义
*
* @return this
* @throws IORuntimeException IO异常
*/
public ExcelWriter flush() throws IORuntimeException {
return flush(this.destFile);
}
/**
* 将Excel Workbook刷出到文件
* 如果用户未自定义输出的文件,将抛出{@link NullPointerException}
*
* @param destFile 写出到的文件
* @return this
* @throws IORuntimeException IO异常
* @since 4.0.6
*/
public ExcelWriter flush(File destFile) throws IORuntimeException {
Assert.notNull(destFile, "[destFile] is null, and you must call setDestFile(File) first or call flush(OutputStream).");
OutputStream out = null;
try {
out = FileUtil.getOutputStream(destFile);
flush(out);
} finally {
IoUtil.close(out);
}
return this;
}
/**
* 将Excel Workbook刷出到输出流
*
* @param out 输出流
* @return this
* @throws IORuntimeException IO异常
*/
public ExcelWriter flush(OutputStream out) throws IORuntimeException {
Assert.isFalse(this.isClosed, "ExcelWriter has been closed!");
try {
this.workbook.write(out);
} catch (IOException e) {
throw new IORuntimeException(e);
}
return this;
}
/**
* 关闭工作簿
* 如果用户设定了目标文件,先写出目标文件后给关闭工作簿
*/
@Override
public void close() {
if (null != this.destFile) {
flush();
}
IoUtil.close(this.workbook);
// 清空对象
this.currentRow = null;
this.styleSet = null;
this.sheet = null;
this.workbook = null;
isClosed = true;
}
// -------------------------------------------------------------------------- Private method start
/**
* 为指定的key列表添加标题别名,如果没有定义key的别名,使用原key
*
* @param keys 键列表
* @return 别名列表
*/
private Collection> aliasHeader(Collection> keys) {
if (MapUtil.isEmpty(this.headerAlias)) {
return keys;
}
final List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy