摘要:緣起最近寫一個程序,需要操作,遂稍微深入了解下從操作,在需要復(fù)制一個的時候,找到三份資料操作,復(fù)制,復(fù)制行,復(fù)制單元格使用操作拷貝導(dǎo)出時設(shè)置單元格類型為數(shù)值類型問題如果在一個的時候,需要把樣式也過去,怎么辦,會怎樣呢如果像第三份資料中所做,
緣起
最近寫一個程序,需要操作excel,遂稍微深入了解下poi從操作,
在需要復(fù)制一個sheet的時候,找到三份資料
poi操作excel,復(fù)制sheet,復(fù)制行,復(fù)制單元格
使用POI操作Excel:Sheet拷貝
POI導(dǎo)出Excel時設(shè)置單元格類型為數(shù)值類型
問題
如果在copy一個cell的時候,需要把樣式也copy過去,怎么辦,會怎樣呢?
如果像第三份資料中所做,會創(chuàng)建太多的style,而這些style又是重復(fù)的 ,毫無意義,數(shù)據(jù)多了還報錯
如果像第二份資料中所做,前提是同一個workbook,也不存在copy style 的問題,只不過是不同的cell引用相同的style而已
如果像第一份資料所做,他自己都注釋掉了,因為報錯
如何解決
思路
相同的book中,不需要copy
不同的book中,styleTable應(yīng)該不一致,copy的時候,在目標(biāo)book中創(chuàng)建,最后返回兩個book中的style index的映射關(guān)系
代碼
暫時沒測試過,寫出來就迫不及待要分享了。
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.Workbook;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* 提供常用excel操作
*
*/
public class ExcelOperationUtil {
private static Logger logger = LoggerFactory.getLogger(ExcelOperationUtil.class);
/**
* 把一個excel中的styleTable復(fù)制到另一個excel中
* @return StyleMapping 兩個文件中styleTable的映射關(guān)系
* @see StyleMapping
*/
public static StyleMapping copyCellStyle(Workbook srcBook, Workbook desBook){
if (null == srcBook || null == desBook) {
throw new ExcelExecption("源excel 或 目標(biāo)excel 不存在");
}
if (srcBook.equals(desBook)) {
throw new ExcelExecption("不要使用此方法在同一個文件中copy style");
}
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
= this.des2srcIndexMapping.length) {
throw new ExcelExecption("索引越界:源文件styleNum=" + this.des2srcIndexMapping.length + " 訪問位置=" + desIndex);
}
return this.des2srcIndexMapping[desIndex];
}
/**
* 根據(jù)源文件的style的index,獲取目標(biāo)文件的style的index
* @param srcIndex 源excel中style的index
* @return desIndex 目標(biāo)excel中style的index
*/
public short desIndex(short srcIndex) {
if (srcIndex < 0 || srcIndex >= this.src2desIndexMapping.length) {
throw new ExcelExecption("索引越界:源文件styleNum=" + this.src2desIndexMapping.length + " 訪問位置=" + srcIndex);
}
return this.src2desIndexMapping[srcIndex];
}
}
}
完善代碼
工作忙了幾天,終于可以回頭把這件事結(jié)束掉了
org.apache.poi
poi
3.17
org.apache.poi
poi-ooxml
3.17
org.apache.poi
poi-scratchpad
3.17
為啥貼出依賴?因為我一開始用3.8版本,如果是xls就可以,如果是xlsx死活不行,最后,換掉版本就可以了。
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;
/**
* 提供常用excel操作
*
*/
public class ExcelOperationUtil {
private static Logger logger = LoggerFactory.getLogger(ExcelOperationUtil.class);
/**
* sheet 復(fù)制,復(fù)制數(shù)據(jù)、如果同一個文件,復(fù)制樣式,不同文件則只復(fù)制數(shù)據(jù)
* 如果是同book中復(fù)制,建議使用workbook中的cloneSheet()方法
*
*
建議用于 不同book間只復(fù)制數(shù)據(jù)
*
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet) {
copySheet(srcSheet, desSheet, true, true, null);
}
/**
* sheet 復(fù)制,如果同一個文件,復(fù)制樣式,不同文件則不復(fù)制
*
*
建議用于 同book中,只復(fù)制樣式,不復(fù)制數(shù)據(jù)
* eg: copySheet(srcSheet, desSheet, false)
*
* @param copyValueFlag 控制是否復(fù)制數(shù)據(jù)
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, boolean copyValueFlag) {
copySheet(srcSheet, desSheet, copyValueFlag, true, null);
}
/**
* sheet 復(fù)制,復(fù)制數(shù)據(jù)、樣式
*
*
建議用于 不同book間復(fù)制,同時復(fù)制數(shù)據(jù)和樣式
* eg: copySheet(srcSheet, desSheet, mapping)
*
* @param mapping 不同文件間復(fù)制時,如果要復(fù)制樣式,必傳,否則不復(fù)制樣式
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, StyleMapping mapping) {
copySheet(srcSheet, desSheet, true, true, mapping);
}
/**
* sheet 復(fù)制,復(fù)制數(shù)據(jù)
*
*
建議用于 同book中,只復(fù)制數(shù)據(jù),不復(fù)制樣式
* 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 復(fù)制, 靈活控制是否控制數(shù)據(jù)、樣式
*
*
不建議直接使用
*
* @param copyValueFlag 控制是否復(fù)制數(shù)據(jù)
* @param copyStyleFlag 控制是否復(fù)制樣式
* @param mapping 不同book中復(fù)制樣式時,必傳
*/
public static void copySheet(Sheet srcSheet, Sheet desSheet, boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
if (srcSheet.getWorkbook() == desSheet.getWorkbook()) {
logger.warn("統(tǒng)一workbook內(nèi)復(fù)制sheet建議使用 workbook的cloneSheet方法");
}
//合并區(qū)域處理
copyMergedRegion(srcSheet, desSheet);
//行復(fù)制
Iterator 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);
//調(diào)整列寬(增量調(diào)整)
if (srcRow.getPhysicalNumberOfCells() > areadlyColunm) {
for (int i = areadlyColunm; i < srcRow.getPhysicalNumberOfCells(); i++) {
desSheet.setColumnWidth(i, srcSheet.getColumnWidth(i));
}
areadlyColunm = srcRow.getPhysicalNumberOfCells();
}
}
}
/**
* 復(fù)制行
*/
public static void copyRow(Row srcRow, Row desRow) {
copyRow(srcRow, desRow, true, true, null);
}
/**
* 復(fù)制行
*/
public static void copyRow(Row srcRow, Row desRow, boolean copyValueFlag) {
copyRow(srcRow, desRow, copyValueFlag, true, null);
}
/**
* 復(fù)制行
*/
public static void copyRow(Row srcRow, Row desRow, StyleMapping mapping) {
copyRow(srcRow, desRow, true, true, mapping);
}
/**
* 復(fù)制行
*/
public static void copyRow(Row srcRow, Row desRow, boolean copyStyleFlag, StyleMapping mapping) {
copyRow(srcRow, desRow, true, copyStyleFlag, mapping);
}
/**
* 復(fù)制行
*/
public static void copyRow(Row srcRow, Row desRow,boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
Iterator it = srcRow.cellIterator();
while (it.hasNext()) {
Cell srcCell = it.next();
Cell desCell = desRow.createCell(srcCell.getColumnIndex());
copyCell(srcCell, desCell, copyValueFlag, copyStyleFlag, mapping);
}
}
/**
* 復(fù)制區(qū)域(合并單元格)
*/
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);
}
}
/**
* 復(fù)制單元格,復(fù)制數(shù)據(jù),如果同文件,復(fù)制樣式,不同文件則不復(fù)制樣式
*/
public static void copyCell(Cell srcCell, Cell desCell) {
copyCell(srcCell, desCell, true, true,null);
}
/**
* 復(fù)制單元格, 如果同文件,復(fù)制樣式,不同文件則不復(fù)制樣式
* @param copyValueFlag 控制是否復(fù)制數(shù)據(jù)
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyValueFlag) {
copyCell(srcCell, desCell, copyValueFlag, true, null);
}
/**
* 復(fù)制單元格,復(fù)制數(shù)據(jù),復(fù)制樣式
* @param mapping 不同文件間復(fù)制時,如果要復(fù)制樣式,必傳,否則不復(fù)制樣式
*/
public static void copyCell(Cell srcCell, Cell desCell, StyleMapping mapping) {
copyCell(srcCell, desCell, true, true, mapping);
}
/**
* 復(fù)制單元格,復(fù)制數(shù)據(jù)
* @param copyStyleFlag 控制是否復(fù)制樣式
* @param mapping 不同文件間復(fù)制時,如果要復(fù)制樣式,必傳,否則不復(fù)制樣式
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyStyleFlag, StyleMapping mapping) {
copyCell(srcCell, desCell, true, copyStyleFlag, mapping);
}
/**
* 復(fù)制單元格
* @param copyValueFlag 控制是否復(fù)制單元格的內(nèi)容
* @param copyStyleFlag 控制是否復(fù)制樣式
* @param mapping 不同文件間復(fù)制時,如果需要連帶樣式復(fù)制,必傳,否則不復(fù)制樣式
*/
public static void copyCell(Cell srcCell, Cell desCell, boolean copyValueFlag, boolean copyStyleFlag, StyleMapping mapping) {
Workbook srcBook = srcCell.getSheet().getWorkbook();
Workbook desBook = desCell.getSheet().getWorkbook();
//復(fù)制樣式
//如果是同一個excel文件內(nèi),連帶樣式一起復(fù)制
if (srcBook == desBook && copyStyleFlag) {
//同文件,復(fù)制引用
desCell.setCellStyle(srcCell.getCellStyle());
} else if (copyStyleFlag) {
//不同文件,通過映射關(guān)系復(fù)制
if (null != mapping) {
short desIndex = mapping.desIndex(srcCell.getCellStyle().getIndex());
desCell.setCellStyle(desBook.getCellStyleAt(desIndex));
}
}
//復(fù)制評論
if (srcCell.getCellComment() != null) {
desCell.setCellComment(srcCell.getCellComment());
}
//復(fù)制內(nèi)容
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復(fù)制到另一個excel中
* 如果是同一個excel文件,就不用復(fù)制styleTable了
* @return StyleMapping 兩個文件中styleTable的映射關(guān)系
* @see StyleMapping
*/
public static StyleMapping copyCellStyle(Workbook srcBook, Workbook desBook){
if (null == srcBook || null == desBook) {
throw new ExcelException("源excel 或 目標(biāo)excel 不存在");
}
if (srcBook.equals(desBook)) {
throw new ExcelException("不要使用此方法在同一個文件中copy style,同一個excel中復(fù)制sheet不需要copy Style");
}
if ((srcBook instanceof HSSFWorkbook && desBook instanceof XSSFWorkbook) ||
(srcBook instanceof XSSFWorkbook && desBook instanceof HSSFWorkbook)) {
throw new ExcelException("不支持在不同的版本的excel中復(fù)制樣式)");
}
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= this.des2srcIndexMapping.length) {
throw new ExcelException("索引越界:源文件styleNum=" + this.des2srcIndexMapping.length + " 訪問位置=" + desIndex);
}
return this.des2srcIndexMapping[desIndex];
}
/**
* 根據(jù)源文件的style的index,獲取目標(biāo)文件的style的index
* @param srcIndex 源excel中style的index
* @return desIndex 目標(biāo)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];
}
}
}
一個自定義異常類
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);
}
}
|
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/71194.html