公司有个需求是根据用户上传的Excel,以及匹配规则,要求根据匹配规则来判断这个excel中的某些sheet页要发送邮箱给不同的用户。
先保存用户上传的excel路径,每次点击发送邮箱的时候。取配置表。
然后根据配置表取出哪些sheet组成新的excel发给哪些用户。
1.使用mq 异步通知发送邮箱
2.mq监听,收到后,多线程消费这条消息里的配置信息。
3.解析excel,拆分sheet到新的excel
4.发送邮箱
1.ExcelCopyUtils
package org.zt.task.report.util;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.ss.util.CellRangeAddress;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.Iterator;
/**
* @author liuhuapeng
* @date 2023/7/5
*/
public class ExcelCopyUtils {
private static Logger logger = LoggerFactory.getLogger(ExcelCopyUtils.class);
/**
* sheet 复制,复制数据、如果同一个文件,复制样式,不同文件则只复制数据
* 如果是同book中复制,建议使用workbook中的cloneSheet()方法
*
*
建议用于 不同book间只复制数据
*
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet) {
copySheet(srcSheet, desSheet, true, true, null);
}
/**
* sheet 复制,如果同一个文件,复制样式,不同文件则不复制
*
*
建议用于 同book中,只复制样式,不复制数据
* eg: copySheet(srcSheet, desSheet, false)
*
* @param copyValueFlag 控制是否复制数据
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, boolean copyValueFlag) {
copySheet(srcSheet, desSheet, copyValueFlag, true, null);
}
/**
* sheet 复制,复制数据、样式
*
*
建议用于 不同book间复制,同时复制数据和样式
* eg: copySheet(srcSheet, desSheet, mapping)
*
* @param mapping 不同文件间复制时,如果要复制样式,必传,否则不复制样式
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, StyleMapping mapping) {
copySheet(srcSheet, desSheet, true, true, mapping);
}
/**
* sheet 复制,复制数据
*
*
建议用于 同book中,只复制数据,不复制样式
* eg: copySheet(srcSheet, desSheet, false, null)
*
* @param srcSheet
* @param desSheet
* @param copyStyleFlag
* @param mapping
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, boolean copyStyleFlag, StyleMapping mapping) {
copySheet(srcSheet, desSheet, true, copyStyleFlag, mapping);
}
/**
* sheet 复制, 灵活控制是否控制数据、样式
*
*
不建议直接使用
*
* @param copyValueFlag 控制是否复制数据
* @param copyStyleFlag 控制是否复制样式
* @param mapping 不同book中复制样式时,必传
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
if (srcSheet.getWorkbook() == desSheet.getWorkbook()) {
logger.warn("统一workbook内复制sheet建议使用 workbook的cloneSheet方法");
}
//合并区域处理
copyMergedRegion(srcSheet, desSheet);
//行复制
Iterator<Row> rowIterator = srcSheet.rowIterator();
int areadlyColunm = 0;
while (rowIterator.hasNext()) {
Row srcRow = rowIterator.next();
Row desRow = desSheet.createRow(srcRow.getRowNum());
copyRow(srcRow, desRow, copyValueFlag, copyStyleFlag, mapping);
//调整列宽(增量调整)
if (srcRow.getPhysicalNumberOfCells() > areadlyColunm) {
for (int i = areadlyColunm; i < srcRow.getPhysicalNumberOfCells(); i++) {
desSheet.setColumnWidth(i, srcSheet.getColumnWidth(i));
}
areadlyColunm = srcRow.getPhysicalNumberOfCells();
}
}
}
/**
* 复制行
*/
public static void copyRow(Row srcRow, Row desRow) {
copyRow(srcRow, desRow, true, true, null);
}
/**
* 复制行
*/
public static void copyRow(Row srcRow, Row desRow, boolean copyValueFlag) {
copyRow(srcRow, desRow, copyValueFlag, true, null);
}
/**
* 复制行
*/
public static void copyRow(Row srcRow, Row desRow, StyleMapping mapping) {
copyRow(srcRow, desRow, true, true, mapping);
}
/**
* 复制行
*/
public static void copyRow(Row srcRow, Row desRow, boolean copyStyleFlag, StyleMapping mapping) {
copyRow(srcRow, desRow, true, copyStyleFlag, mapping);
}
/**
* 复制行
*/
public static void copyRow(Row srcRow, Row desRow,boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
Iterator<Cell> it = srcRow.cellIterator();
while (it.hasNext()) {
Cell srcCell = it.next();
Cell desCell = desRow.createCell(srcCell.getColumnIndex());
copyCell(srcCell, desCell, copyValueFlag, copyStyleFlag, mapping);
}
}
/**
* 复制区域(合并单元格)
*/
public static void copyMergedRegion(Sheet srcSheet, Sheet desSheet) {
int sheetMergerCount = srcSheet.getNumMergedRegions();
for (int i = 0; i < sheetMergerCount; i++) {
desSheet.addMergedRegion(srcSheet.getMergedRegion(i));
CellRangeAddress cellRangeAddress = srcSheet.getMergedRegion(i);
}
}
/**
* 复制单元格,复制数据,如果同文件,复制样式,不同文件则不复制样式
*/
public static void copyCell(Cell srcCell, Cell desCell) {
copyCell(srcCell, desCell, true, true,null);
}
/**
* 复制单元格, 如果同文件,复制样式,不同文件则不复制样式
* @param copyValueFlag 控制是否复制数据
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyValueFlag) {
copyCell(srcCell, desCell, copyValueFlag, true, null);
}
/**
* 复制单元格,复制数据,复制样式
* @param mapping 不同文件间复制时,如果要复制样式,必传,否则不复制样式
*/
public static void copyCell(Cell srcCell, Cell desCell, StyleMapping mapping) {
copyCell(srcCell, desCell, true, true, mapping);
}
/**
* 复制单元格,复制数据
* @param copyStyleFlag 控制是否复制样式
* @param mapping 不同文件间复制时,如果要复制样式,必传,否则不复制样式
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyStyleFlag, StyleMapping mapping) {
copyCell(srcCell, desCell, true, copyStyleFlag, mapping);
}
/**
* 复制单元格
* @param copyValueFlag 控制是否复制单元格的内容
* @param copyStyleFlag 控制是否复制样式
* @param mapping 不同文件间复制时,如果需要连带样式复制,必传,否则不复制样式
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
Workbook srcBook = srcCell.getSheet().getWorkbook();
Workbook desBook = desCell.getSheet().getWorkbook();
//复制样式
//如果是同一个excel文件内,连带样式一起复制
if (srcBook == desBook && copyStyleFlag) {
//同文件,复制引用
desCell.setCellStyle(srcCell.getCellStyle());
} else if (copyStyleFlag) {
//不同文件,通过映射关系复制
if (null != mapping) {
short desIndex = mapping.desIndex(srcCell.getCellStyle().getIndex());
desCell.setCellStyle(desBook.getCellStyleAt(desIndex));
}
}
//复制评论
if (srcCell.getCellComment() != null) {
desCell.setCellComment(srcCell.getCellComment());
}
//复制内容
desCell.setCellType(srcCell.getCellTypeEnum());
if (copyValueFlag) {
switch (srcCell.getCellTypeEnum()) {
case STRING:
desCell.setCellValue(srcCell.getStringCellValue());
break;
case NUMERIC:
desCell.setCellValue(srcCell.getNumericCellValue());
break;
case FORMULA:
desCell.setCellFormula(srcCell.getCellFormula());
break;
case BOOLEAN:
desCell.setCellValue(srcCell.getBooleanCellValue());
break;
case ERROR:
desCell.setCellValue(srcCell.getErrorCellValue());
break;
case BLANK:
//nothing to do
break;
default:
break;
}
}
}
/**
* 把一个excel中的styleTable复制到另一个excel中
* 如果是同一个excel文件,就不用复制styleTable了
* @return StyleMapping 两个文件中styleTable的映射关系
* @see StyleMapping
*/
public static StyleMapping copyCellStyle(Workbook srcBook, Workbook desBook){
if (null == srcBook || null == desBook) {
throw new ExcelException("源excel 或 目标excel 不存在");
}
if (srcBook.equals(desBook)) {
throw new ExcelException("不要使用此方法在同一个文件中copy style,同一个excel中复制sheet不需要copy Style");
}
if ((srcBook instanceof HSSFWorkbook && desBook instanceof XSSFWorkbook) ||
(srcBook instanceof XSSFWorkbook && desBook instanceof HSSFWorkbook)) {
throw new ExcelException("不支持在不同的版本的excel中复制样式)");
}
logger.debug("src中style number:{}, des中style number:{}", srcBook.getNumCellStyles(), desBook.getNumCellStyles());
short[] src2des = new short[srcBook.getNumCellStyles()];
short[] des2src = new short[desBook.getNumCellStyles() + srcBook.getNumCellStyles()];
for(short i=0;i<srcBook.getNumCellStyles();i++){
//建立双向映射
CellStyle srcStyle = srcBook.getCellStyleAt(i);
CellStyle desStyle = desBook.createCellStyle();
src2des[srcStyle.getIndex()] = desStyle.getIndex();
des2src[desStyle.getIndex()] = srcStyle.getIndex();
//复制样式
desStyle.cloneStyleFrom(srcStyle);
}
return new StyleMapping(des2src, src2des);
}
/**
* 存放两个excel文件中的styleTable的映射关系,以便于在复制表格时,在目标文件中获取到对应的样式
*/
public static class StyleMapping {
/**
*
*/
private short[] des2srcIndexMapping;
/**
*
*/
private short[] src2desIndexMapping;
/**
* 不允许其他类创建此类型对象
*/
private StyleMapping() {
}
public StyleMapping(short[] des2srcIndexMapping, short[] src2desIndexMapping) {
this.des2srcIndexMapping = des2srcIndexMapping;
this.src2desIndexMapping = src2desIndexMapping;
}
public short srcIndex(short desIndex) {
if (desIndex < 0 || desIndex >= this.des2srcIndexMapping.length) {
throw new ExcelException("索引越界:源文件styleNum=" + this.des2srcIndexMapping.length + " 访问位置=" + desIndex);
}
return this.des2srcIndexMapping[desIndex];
}
/**
* 根据源文件的style的index,获取目标文件的style的index
* @param srcIndex 源excel中style的index
* @return desIndex 目标excel中style的index
*/
public short desIndex(short srcIndex) {
if (srcIndex < 0 || srcIndex >= this.src2desIndexMapping.length) {
throw new ExcelException("索引越界:源文件styleNum=" + this.src2desIndexMapping.length + " 访问位置=" + srcIndex);
}
return this.src2desIndexMapping[srcIndex];
}
}
}
2.自定义异常ExcelExpetion
package org.zt.task.report.util;
/**
* @author liuhuapeng
* @date 2023/7/5
*/
public class ExcelException extends RuntimeException{
public ExcelException() {
}
public ExcelException(String message) {
super(message);
}
public ExcelException(String message, Throwable cause) {
super(message, cause);
}
public ExcelException(Throwable cause) {
super(cause);
}
public ExcelException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
super(message, cause, enableSuppression, writableStackTrace);
}
}
3.依赖
因为我们项目引入了easyexcel,他会自己引入poi的包
4…使用的话,自己研究一下就可以了。
这个方法也是我从网上别的大佬那里看到的,亲测可用。
大佬的文章链接找不到了,有看到的小伙伴踢我一脚,我再补充上。
站在巨人的肩膀上前行~