From 95496f1d46c68884d329ba44e87bc2889d429479 Mon Sep 17 00:00:00 2001 From: liujifei <327714@163.com> Date: Mon, 1 Jul 2019 12:41:29 +0800 Subject: [PATCH 1/2] =?UTF-8?q?1=20=E6=94=AF=E6=8C=81=E8=AE=BE=E7=BD=AE?= =?UTF-8?q?=E8=BF=9E=E7=BB=ADn=E4=B8=AA=E7=A9=BA=E7=99=BD=E8=A1=8C?= =?UTF-8?q?=E6=89=8D=E6=8E=A8=E5=87=BA=E8=AF=BB=E5=8F=96excel=202=20?= =?UTF-8?q?=E6=94=AF=E6=8C=81=E8=87=AA=E5=AE=9A=E4=B9=89=E5=AD=97=E6=AE=B5?= =?UTF-8?q?=E5=AF=B9=E5=BA=94excel=E5=88=97=E5=BA=8F=E5=8F=B7=E3=80=82?= =?UTF-8?q?=E4=B8=8D=E8=AE=BE=E7=BD=AE=E5=88=97=E5=BA=8F=E5=8F=B7=E5=88=99?= =?UTF-8?q?=E6=8C=89=E7=85=A7=E5=AD=97=E6=AE=B5=E5=AE=9A=E4=B9=89=E9=A1=BA?= =?UTF-8?q?=E5=BA=8F=E8=AE=BE=E7=BD=AEexcel=E5=88=97=E5=BA=8F=E5=8F=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- pom.xml | 2 +- src/main/java/com/excel/poi/ExcelBoot.java | 70 +++++++-- .../com/excel/poi/annotation/ImportField.java | 12 ++ .../java/com/excel/poi/common/Constant.java | 1 + .../excel/poi/entity/ExcelPropertyEntity.java | 6 + .../java/com/excel/poi/excel/ExcelReader.java | 141 ++++++++++++------ .../poi/factory/ExcelMappingFactory.java | 2 + .../java/com/excel/poi/utils/ExcelUtils.java | 60 ++++++++ src/test/java/com/excel/poi/test/ExcelVO.java | 67 +++++++++ src/test/java/com/excel/poi/test/Test.java | 50 +++++++ src/test/java/com/excel/poi/test/Test2.java | 49 ++++++ .../com/excel/poi/utils/ExcelUtilsTest.java | 67 +++++++++ 12 files changed, 464 insertions(+), 63 deletions(-) create mode 100644 src/main/java/com/excel/poi/utils/ExcelUtils.java create mode 100644 src/test/java/com/excel/poi/test/ExcelVO.java create mode 100644 src/test/java/com/excel/poi/test/Test.java create mode 100644 src/test/java/com/excel/poi/test/Test2.java create mode 100644 src/test/java/com/excel/poi/utils/ExcelUtilsTest.java diff --git a/pom.xml b/pom.xml index ce22b08..80f9084 100644 --- a/pom.xml +++ b/pom.xml @@ -6,7 +6,7 @@ io.github.magic-core excel-boot - 2.0 + 2.01 org.sonatype.oss diff --git a/src/main/java/com/excel/poi/ExcelBoot.java b/src/main/java/com/excel/poi/ExcelBoot.java index c29f43b..c9e3d4f 100644 --- a/src/main/java/com/excel/poi/ExcelBoot.java +++ b/src/main/java/com/excel/poi/ExcelBoot.java @@ -28,6 +28,7 @@ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.URLEncoder; +import java.util.List; import javax.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; import org.apache.poi.openxml4j.exceptions.OpenXML4JException; @@ -48,6 +49,7 @@ public class ExcelBoot { private Integer rowAccessWindowSize; private Integer recordCountPerSheet; private Boolean openAutoColumWidth; + private Integer maxContiguousBlankRowCount = Constant.MAX_ROW_COUNT_CONTIGUOUS_BLANK; /** * 导入构造器 @@ -56,7 +58,17 @@ public class ExcelBoot { * @param excelClass */ protected ExcelBoot(InputStream inputStream, Class excelClass) { - this(null, null, inputStream, null, excelClass, null, null, null, null); + this(null, null, inputStream, null, excelClass, null, null, null, null, null); + } + + /** + * 导入构造器 + * + * @param inputStream + * @param excelClass + */ + protected ExcelBoot(InputStream inputStream, Class excelClass, Integer maxContiguousBlankRowCount) { + this(null, null, inputStream, null, excelClass, null, null, null, null, maxContiguousBlankRowCount); } /** @@ -67,7 +79,7 @@ public class ExcelBoot { * @param excelClass */ protected ExcelBoot(OutputStream outputStream, String fileName, Class excelClass) { - this(null, outputStream, null, fileName, excelClass, Constant.DEFAULT_PAGE_SIZE, Constant.DEFAULT_ROW_ACCESS_WINDOW_SIZE, Constant.DEFAULT_RECORD_COUNT_PEER_SHEET, Constant.OPEN_AUTO_COLUM_WIDTH); + this(null, outputStream, null, fileName, excelClass, Constant.DEFAULT_PAGE_SIZE, Constant.DEFAULT_ROW_ACCESS_WINDOW_SIZE, Constant.DEFAULT_RECORD_COUNT_PEER_SHEET, Constant.OPEN_AUTO_COLUM_WIDTH, Constant.MAX_ROW_COUNT_CONTIGUOUS_BLANK); } /** @@ -78,7 +90,7 @@ public class ExcelBoot { * @param excelClass */ protected ExcelBoot(HttpServletResponse response, String fileName, Class excelClass) { - this(response, null, null, fileName, excelClass, Constant.DEFAULT_PAGE_SIZE, Constant.DEFAULT_ROW_ACCESS_WINDOW_SIZE, Constant.DEFAULT_RECORD_COUNT_PEER_SHEET, Constant.OPEN_AUTO_COLUM_WIDTH); + this(response, null, null, fileName, excelClass, Constant.DEFAULT_PAGE_SIZE, Constant.DEFAULT_ROW_ACCESS_WINDOW_SIZE, Constant.DEFAULT_RECORD_COUNT_PEER_SHEET, Constant.OPEN_AUTO_COLUM_WIDTH, Constant.MAX_ROW_COUNT_CONTIGUOUS_BLANK); } /** @@ -95,7 +107,7 @@ public class ExcelBoot { * @param openAutoColumWidth */ protected ExcelBoot(HttpServletResponse response, OutputStream outputStream, InputStream inputStream - , String fileName, Class excelClass, Integer pageSize, Integer rowAccessWindowSize, Integer recordCountPerSheet, Boolean openAutoColumWidth) { + , String fileName, Class excelClass, Integer pageSize, Integer rowAccessWindowSize, Integer recordCountPerSheet, Boolean openAutoColumWidth, Integer maxContiguousBlankRowCount) { this.httpServletResponse = response; this.outputStream = outputStream; this.inputStream = inputStream; @@ -105,6 +117,7 @@ public class ExcelBoot { this.rowAccessWindowSize = rowAccessWindowSize; this.recordCountPerSheet = recordCountPerSheet; this.openAutoColumWidth = openAutoColumWidth; + this.maxContiguousBlankRowCount = maxContiguousBlankRowCount; } /** @@ -146,7 +159,7 @@ public class ExcelBoot { public static ExcelBoot ExportBuilder(HttpServletResponse response, String fileName, Class excelClass, Integer pageSize, Integer rowAccessWindowSize, Integer recordCountPerSheet, Boolean openAutoColumWidth) { return new ExcelBoot(response, null, null - , fileName, excelClass, pageSize, rowAccessWindowSize, recordCountPerSheet, openAutoColumWidth); + , fileName, excelClass, pageSize, rowAccessWindowSize, recordCountPerSheet, openAutoColumWidth, null); } /** @@ -164,7 +177,7 @@ public class ExcelBoot { public static ExcelBoot ExportBuilder(OutputStream outputStream, String fileName, Class excelClass, Integer pageSize , Integer rowAccessWindowSize, Integer recordCountPerSheet, Boolean openAutoColumWidth) { return new ExcelBoot(null, outputStream, null - , fileName, excelClass, pageSize, rowAccessWindowSize, recordCountPerSheet, openAutoColumWidth); + , fileName, excelClass, pageSize, rowAccessWindowSize, recordCountPerSheet, openAutoColumWidth, null); } /** @@ -178,12 +191,22 @@ public class ExcelBoot { return new ExcelBoot(inputStreamm, clazz); } + /** + * 导入Excel文件数据 + * + * @param inputStreamm + * @param clazz + * @return + */ + public static ExcelBoot ImportBuilder(InputStream inputStreamm, Class clazz, int maxContiguousBlankRowCount) { + return new ExcelBoot(inputStreamm, clazz, maxContiguousBlankRowCount); + } + /** * 用于浏览器导出 * * @param param * @param exportFunction - * @param ExportFunction * @param * @param */ @@ -211,7 +234,6 @@ public class ExcelBoot { * 通过OutputStream导出excel文件,一般用于异步导出大Excel文件到本地路径 * * @param param - * @param ExportFunction * @param exportFunction * @param * @param @@ -237,7 +259,6 @@ public class ExcelBoot { * * @param param * @param exportFunction - * @param ExportFunction * @param * @param * @return @@ -263,7 +284,6 @@ public class ExcelBoot { * * @param param * @param exportFunction - * @param ExportFunction * @param * @param */ @@ -289,7 +309,6 @@ public class ExcelBoot { * 通过OutputStream分sheet导出excel文件,一般用于异步导出大Excel文件到本地路径 * * @param param - * @param ExportFunction * @param exportFunction * @param * @param @@ -316,7 +335,6 @@ public class ExcelBoot { * * @param param * @param exportFunction - * @param ExportFunction * @param * @param * @return @@ -340,7 +358,6 @@ public class ExcelBoot { /** * 导出-导入模板 * - * @param data * @throws Exception */ public void exportTemplate() { @@ -369,7 +386,6 @@ public class ExcelBoot { /** * 导入excel全部sheet * - * @param inputStream * @param importFunction * @throws OpenXML4JException * @throws SAXException @@ -393,6 +409,32 @@ public class ExcelBoot { } + /** + * 导入excel指定sheet + * + * @param importFunction + * @throws OpenXML4JException + * @throws SAXException + * @throws IOException + */ + public void importExcel(ImportFunction importFunction, List sheetIndexs) { + try { + if (importFunction == null) { + throw new ExcelBootException("excelReadHandler参数为空!"); + } + if (inputStream == null) { + throw new ExcelBootException("inputStream参数为空!"); + } + + ExcelEntity excelMapping = ExcelMappingFactory.loadImportExcelClass(excelClass); + ExcelReader excelReader = new ExcelReader(excelClass, excelMapping, 0, maxContiguousBlankRowCount, importFunction); + excelReader.process(inputStream, sheetIndexs); + } catch (Exception e) { + throw new ExcelBootException(e); + } + + } + private SXSSFWorkbook commonSingleSheet(R param, ExportFunction exportFunction) throws Exception { verifyParams(); ExcelEntity excelMapping = ExcelMappingFactory.loadExportExcelClass(excelClass, fileName); diff --git a/src/main/java/com/excel/poi/annotation/ImportField.java b/src/main/java/com/excel/poi/annotation/ImportField.java index a23d60b..47c346e 100644 --- a/src/main/java/com/excel/poi/annotation/ImportField.java +++ b/src/main/java/com/excel/poi/annotation/ImportField.java @@ -68,4 +68,16 @@ public @interface ImportField { * @return */ int roundingMode() default BigDecimal.ROUND_HALF_EVEN; + + /** + * excel列序号 从0开始 优先取序号,无序号则取序号名,前述情况外取类字段申明顺序 默认:-1 + * @return + */ + int excelColumnIndex() default -1; + + /** + * excel列序号名称,即excel显示的列名 从A开始 默认:NONE + * @return + */ + String excelColumnIndexName() default "NONE"; } diff --git a/src/main/java/com/excel/poi/common/Constant.java b/src/main/java/com/excel/poi/common/Constant.java index 6c177bf..2be114a 100644 --- a/src/main/java/com/excel/poi/common/Constant.java +++ b/src/main/java/com/excel/poi/common/Constant.java @@ -47,4 +47,5 @@ public final class Constant { public static final int CHINESES_ATUO_SIZE_COLUMN_WIDTH_MIN = 15; public static final int MAX_RECORD_COUNT_PEER_SHEET = 1000000; + public static final int MAX_ROW_COUNT_CONTIGUOUS_BLANK = 0; } diff --git a/src/main/java/com/excel/poi/entity/ExcelPropertyEntity.java b/src/main/java/com/excel/poi/entity/ExcelPropertyEntity.java index 8b0ae1a..f251a30 100644 --- a/src/main/java/com/excel/poi/entity/ExcelPropertyEntity.java +++ b/src/main/java/com/excel/poi/entity/ExcelPropertyEntity.java @@ -79,4 +79,10 @@ public class ExcelPropertyEntity { * @return 是否必填 */ private Boolean required; + + /** + * excel列序号 从0开始 默认:0 + * @return + */ + private int excelColumnIndex; } diff --git a/src/main/java/com/excel/poi/excel/ExcelReader.java b/src/main/java/com/excel/poi/excel/ExcelReader.java index 83c1fe5..ac3856a 100644 --- a/src/main/java/com/excel/poi/excel/ExcelReader.java +++ b/src/main/java/com/excel/poi/excel/ExcelReader.java @@ -32,11 +32,10 @@ import java.io.InputStream; import java.lang.reflect.Field; import java.math.BigDecimal; import java.text.ParseException; -import java.util.ArrayList; -import java.util.Date; -import java.util.Iterator; -import java.util.List; +import java.util.*; import java.util.concurrent.ExecutionException; + +import com.excel.poi.utils.ExcelUtils; import lombok.extern.slf4j.Slf4j; import org.apache.poi.openxml4j.exceptions.OpenXML4JException; import org.apache.poi.openxml4j.opc.OPCPackage; @@ -55,6 +54,8 @@ import org.xml.sax.helpers.XMLReaderFactory; */ @Slf4j public class ExcelReader extends DefaultHandler { + private Integer maxContiguousBlankRowCount = 0; + private Integer contiguousBlankRowCount = 0; private Integer currentSheetIndex = -1; private Integer currentRowIndex = 0; private Integer excelCurrentCellIndex = 0; @@ -68,7 +69,7 @@ public class ExcelReader extends DefaultHandler { private ExcelEntity excelMapping; private ImportFunction importFunction; private Class excelClass; - private List cellsOnRow = new ArrayList(); + private Map cellsOnRow = new HashMap(); private Integer beginReadRowIndex; private Integer dataCurrentCellIndex = -1; @@ -76,16 +77,29 @@ public class ExcelReader extends DefaultHandler { public ExcelReader(Class entityClass, ExcelEntity excelMapping, ImportFunction importFunction) { - this(entityClass, excelMapping, 1, importFunction); + this(entityClass, excelMapping, 1, Constant.MAX_ROW_COUNT_CONTIGUOUS_BLANK, importFunction); + } + + public ExcelReader(Class entityClass, + ExcelEntity excelMapping, + Integer beginReadRowIndex, + ImportFunction importFunction) { + this.excelClass = entityClass; + this.excelMapping = excelMapping; + this.beginReadRowIndex = beginReadRowIndex; + this.maxContiguousBlankRowCount = Constant.MAX_ROW_COUNT_CONTIGUOUS_BLANK; + this.importFunction = importFunction; } public ExcelReader(Class entityClass, ExcelEntity excelMapping, Integer beginReadRowIndex, + Integer maxContiguousBlankRowCount, ImportFunction importFunction) { this.excelClass = entityClass; this.excelMapping = excelMapping; this.beginReadRowIndex = beginReadRowIndex; + this.maxContiguousBlankRowCount = maxContiguousBlankRowCount; this.importFunction = importFunction; } @@ -127,6 +141,44 @@ public class ExcelReader extends DefaultHandler { } } + public void process(InputStream in, List sheetIndexs) + throws IOException, OpenXML4JException, SAXException { + OPCPackage opcPackage = null; + InputStream sheet = null; + InputSource sheetSource; + try { + opcPackage = OPCPackage.open(in); + XSSFReader xssfReader = new XSSFReader(opcPackage); + XMLReader parser = this.fetchSheetParser(xssfReader.getSharedStringsTable()); + + for(Integer sheetIndex : sheetIndexs) { + contiguousBlankRowCount = 0; + currentRowIndex = 0; + currentSheetIndex = sheetIndex; + try { + sheet = xssfReader.getSheet("rId" + (currentSheetIndex + 1)); + sheetSource = new InputSource(sheet); + try { + log.info("开始读取第{}个Sheet!", currentSheetIndex + 1); + parser.parse(sheetSource); + } catch (AllEmptyRowException e) { + log.warn(e.getMessage()); + } catch (Exception e) { + throw new ExcelBootException(e, "第{}个Sheet,第{}行,第{}列,系统发生异常! ", currentSheetIndex + 1, currentRowIndex + 1, dataCurrentCellIndex + 1); + } + } finally { + if (sheet != null) { + sheet.close(); + } + } + } + } finally { + if (opcPackage != null) { + opcPackage.close(); + } + } + } + /** * 获取sharedStrings.xml文件的XMLReader对象 * @@ -192,27 +244,20 @@ public class ExcelReader extends DefaultHandler { } if (!currentCellLocation.equals(previousCellLocation) && currentRowIndex != 0) { for (int i = 0; i < countNullCell(currentCellLocation, previousCellLocation); i++) { - cellsOnRow.add(excelCurrentCellIndex, ""); + cellsOnRow.put(excelCurrentCellIndex, ""); excelCurrentCellIndex++; } } - if (currentRowIndex != 0 || !"".equals(currentCellValue.trim())) { + if (!"".equals(currentCellValue.trim())) { String value = this.getCellValue(currentCellValue.trim()); - cellsOnRow.add(excelCurrentCellIndex, value); + cellsOnRow.put(excelCurrentCellIndex, value); excelCurrentCellIndex++; } } else if (Constant.ROW.equals(name)) { - if (currentRowIndex == 0) { - endCellLocation = currentCellLocation; - int propertySize = excelMapping.getPropertyList().size(); - if (cellsOnRow.size() != propertySize) { - throw new ExcelBootException("Excel有效列数不等于标注注解的属性数量!Excel列数:{},标注注解的属性数量:{}", cellsOnRow.size(), propertySize); - } - } if (null != endCellLocation) { for (int i = 0; i <= countNullCell(endCellLocation, currentCellLocation); i++) { - cellsOnRow.add(excelCurrentCellIndex, ""); + cellsOnRow.put(excelCurrentCellIndex, ""); excelCurrentCellIndex++; } } @@ -238,7 +283,6 @@ public class ExcelReader extends DefaultHandler { * 根据c节点的s属性获取单元格样式,去styles.xml文件找相应样式 * * @param cellType xml中单元格格式属性 - * @param cellStyleStr xml中样式属性 */ private void setCellType(String cellType) { if ("inlineStr".equals(cellType)) { @@ -265,23 +309,45 @@ public class ExcelReader extends DefaultHandler { } } + private int getMaxColumnNumber() { + List propertyList = excelMapping.getPropertyList(); + int maxIndex = -1; + for (int i = 0; i < propertyList.size(); i++) { + if (propertyList.get(i).getExcelColumnIndex() > maxIndex) { + maxIndex = propertyList.get(i).getExcelColumnIndex(); + } + } + return maxIndex + 1; + } + private void assembleData() throws Exception { if (currentRowIndex >= beginReadRowIndex) { List propertyList = excelMapping.getPropertyList(); - for (int i = 0; i < propertyList.size() - cellsOnRow.size(); i++) { - cellsOnRow.add(i, ""); + for (int i = 0; i < getMaxColumnNumber(); i++) { + if(cellsOnRow.get(i) != null) { + continue; + } + cellsOnRow.put(i, ""); } if (isAllEmptyRowData()) { + ++contiguousBlankRowCount; + } else { + contiguousBlankRowCount = 0; + } + if (isAllEmptyRowData() && contiguousBlankRowCount > maxContiguousBlankRowCount) { throw new AllEmptyRowException("第{}行为空行,第{}个Sheet导入结束!", currentRowIndex + 1, currentSheetIndex + 1); } + if (isAllEmptyRowData()) { + return; + } Object entity = excelClass.newInstance(); ErrorEntity errorEntity = ErrorEntity.builder().build(); for (int i = 0; i < propertyList.size(); i++) { - dataCurrentCellIndex = i; - Object cellValue = cellsOnRow.get(i); + dataCurrentCellIndex = propertyList.get(i).getExcelColumnIndex(); + Object cellValue = cellsOnRow.get(propertyList.get(i).getExcelColumnIndex()); ExcelPropertyEntity property = propertyList.get(i); - errorEntity = checkCellValue(i, property, cellValue); + errorEntity = checkCellValue(propertyList.get(i).getExcelColumnIndex(), property, cellValue); if (errorEntity.getErrorMessage() != null) { break; } @@ -301,8 +367,8 @@ public class ExcelReader extends DefaultHandler { private boolean isAllEmptyRowData() { int emptyCellCount = 0; - for (Object cellData : cellsOnRow) { - if (StringUtil.isBlank(cellData)) { + for (Map.Entry cellData : cellsOnRow.entrySet()) { + if (StringUtil.isBlank(cellData.getValue())) { emptyCellCount++; } } @@ -387,8 +453,8 @@ public class ExcelReader extends DefaultHandler { String xfdA = refA.replaceAll("\\d+", ""); String xfdB = refB.replaceAll("\\d+", ""); - xfdA = fillChar(xfdA, 3, '@', true); - xfdB = fillChar(xfdB, 3, '@', true); + xfdA = ExcelUtils.fillChar(xfdA, 3, '@', true); + xfdB = ExcelUtils.fillChar(xfdB, 3, '@', true); char[] letterA = xfdA.toCharArray(); char[] letterB = xfdB.toCharArray(); @@ -396,27 +462,6 @@ public class ExcelReader extends DefaultHandler { return res - 1; } - private String fillChar(String str, int len, char let, boolean isPre) { - int lenA = str.length(); - if (lenA < len) { - if (isPre) { - StringBuilder strBuilder = new StringBuilder(str); - for (int i = 0; i < (len - lenA); i++) { - strBuilder.insert(0, let); - } - str = strBuilder.toString(); - } else { - StringBuilder strBuilder = new StringBuilder(str); - for (int i = 0; i < (len - lenA); i++) { - strBuilder.append(let); - } - str = strBuilder.toString(); - } - } - return str; - } - - /** * 单元格中的数据可能的数据类型 */ diff --git a/src/main/java/com/excel/poi/factory/ExcelMappingFactory.java b/src/main/java/com/excel/poi/factory/ExcelMappingFactory.java index 4d2cab7..76324a2 100644 --- a/src/main/java/com/excel/poi/factory/ExcelMappingFactory.java +++ b/src/main/java/com/excel/poi/factory/ExcelMappingFactory.java @@ -16,6 +16,7 @@ */ package com.excel.poi.factory; +import com.excel.poi.utils.ExcelUtils; import com.excel.poi.annotation.ExportField; import com.excel.poi.annotation.ImportField; import com.excel.poi.entity.ExcelEntity; @@ -54,6 +55,7 @@ public class ExcelMappingFactory { .regexMessage(importField.regexMessage().trim()) .scale(importField.scale()) .roundingMode(importField.roundingMode()) + .excelColumnIndex(ExcelUtils.getExcelColumnIndex(importField.excelColumnIndex(), importField.excelColumnIndexName(), propertyList.size())) .build(); propertyList.add(excelPropertyEntity); } diff --git a/src/main/java/com/excel/poi/utils/ExcelUtils.java b/src/main/java/com/excel/poi/utils/ExcelUtils.java new file mode 100644 index 0000000..0eb97f2 --- /dev/null +++ b/src/main/java/com/excel/poi/utils/ExcelUtils.java @@ -0,0 +1,60 @@ +package com.excel.poi.utils; + +import com.excel.poi.exception.ExcelBootException; + +import java.util.HashMap; +import java.util.Map; + +public class ExcelUtils { + + public final static Map CHAR_NUMBER = getChar2Number(); + private static Map getChar2Number() { + Map char2Number = new HashMap(); + char2Number.put('0', 0); + for(int i=0;i<26;i++) { + char2Number.put((char) (i+65), i+1); + } + return char2Number; + } + + + public static int getExcelColumnIndex(int columnIndex, String columnIndexName, int defaultIndex) { + if(columnIndex >= 0) { + return columnIndex; + } + if(null == columnIndexName || "".equals(columnIndexName) || !columnIndexName.trim().matches("^[a-zA-Z]{1,3}$")) { + return defaultIndex; + } + columnIndexName = columnIndexName.trim().toUpperCase(); + columnIndexName = fillChar(columnIndexName, 3, '0', true); + char[] columnIndexNameParts = columnIndexName.toCharArray(); + if(columnIndexNameParts[0] >= 'X' && columnIndexNameParts[1] >= 'F' && columnIndexNameParts[2] > 'D') { + throw new ExcelBootException("Excel 2007及之后版本列最大为XFD(16383),当前设置为" + columnIndexName); + } + return CHAR_NUMBER.get(columnIndexNameParts[0]) * 26 * 26 + + CHAR_NUMBER.get(columnIndexNameParts[1]) * 26 + + CHAR_NUMBER.get(columnIndexNameParts[2]) - + 1; + } + + public static String fillChar(String str, int len, char let, boolean isPre) { + int lenA = str.length(); + if (lenA < len) { + if (isPre) { + StringBuilder strBuilder = new StringBuilder(str); + for (int i = 0; i < (len - lenA); i++) { + strBuilder.insert(0, let); + } + str = strBuilder.toString(); + } else { + StringBuilder strBuilder = new StringBuilder(str); + for (int i = 0; i < (len - lenA); i++) { + strBuilder.append(let); + } + str = strBuilder.toString(); + } + } + return str; + } + +} diff --git a/src/test/java/com/excel/poi/test/ExcelVO.java b/src/test/java/com/excel/poi/test/ExcelVO.java new file mode 100644 index 0000000..01da8a0 --- /dev/null +++ b/src/test/java/com/excel/poi/test/ExcelVO.java @@ -0,0 +1,67 @@ +package com.excel.poi.test; + +import com.excel.poi.annotation.ImportField; + +public class ExcelVO { + + @ImportField(regex = "65|65\\d{9}", regexMessage="非批号" , required = true) + private String col_1; + @ImportField(regex = "[\\s\\S]*", regexMessage="列2校验失败", required = true) + private String col_2; + @ImportField(regex = "((CF|cf|Cf|cF)\\d{4})|(\\u4e00\\u53e3\\u4ef7)", regexMessage="列3校验失败", required = true) + private String col_3; + @ImportField(regex = "\\d+(\\.\\d{1,2})?", regexMessage="金额格式不正确", required = true) + private String col_4; + @ImportField(regex = "[\\s\\S]*", regexMessage="列5校验失败") + private String col_5; + @ImportField(regex = "[\\s\\S]*", regexMessage="列6校验失败") + private String col_6; + + public String getCol_1() { + return col_1; + } + + public void setCol_1(String col_1) { + this.col_1 = col_1; + } + + public String getCol_2() { + return col_2; + } + + public void setCol_2(String col_2) { + this.col_2 = col_2; + } + + public String getCol_3() { + return col_3; + } + + public void setCol_3(String col_3) { + this.col_3 = col_3; + } + + public String getCol_4() { + return col_4; + } + + public void setCol_4(String col_4) { + this.col_4 = col_4; + } + + public String getCol_5() { + return col_5; + } + + public void setCol_5(String col_5) { + this.col_5 = col_5; + } + + public String getCol_6() { + return col_6; + } + + public void setCol_6(String col_6) { + this.col_6 = col_6; + } +} diff --git a/src/test/java/com/excel/poi/test/Test.java b/src/test/java/com/excel/poi/test/Test.java new file mode 100644 index 0000000..87618cb --- /dev/null +++ b/src/test/java/com/excel/poi/test/Test.java @@ -0,0 +1,50 @@ +package com.excel.poi.test; + +import com.excel.poi.ExcelBoot; +import com.excel.poi.entity.ErrorEntity; +import com.excel.poi.function.ImportFunction; + +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +public class Test { + public static void main(String... args) throws FileNotFoundException { + + Map batchCounterMap = new HashMap(); + batchCounterMap.put("counter", 0); + batchCounterMap.put("bad", 0); + + final List batchCodes = new ArrayList(); + final StringBuffer error = new StringBuffer(); + ExcelBoot.ImportBuilder(new FileInputStream("C:\\TDDownload\\点价资源数据 (6) - 副本.xlsx"), ExcelVO.class) + .importExcel(new ImportFunction() { + + /** + * @param sheetIndex 当前执行的Sheet的索引, 从1开始 + * @param rowIndex 当前执行的行数, 从1开始 + * @param inEntity Excel行数据的实体 + */ + @Override + public void onProcess(int sheetIndex, int rowIndex, ExcelVO inEntity) { + //将读取到Excel中每一行的inEntity数据进行自定义处理 + //如果该行数据发生问题,将不会走本方法,而会走onError方法 + batchCodes.add(inEntity.getCol_1()); + } + + /** + * @param errorEntity 错误信息实体 + */ + @Override + public void onError(ErrorEntity errorEntity) { + //将每条数据非空和正则校验后的错误信息errorEntity进行自定义处理 + error.append(errorEntity.getRowIndex()+1).append("行数据处理错误,\r\n"); + } + }); + System.out.println(batchCodes); + System.out.println(error); + } +} diff --git a/src/test/java/com/excel/poi/test/Test2.java b/src/test/java/com/excel/poi/test/Test2.java new file mode 100644 index 0000000..8227ae0 --- /dev/null +++ b/src/test/java/com/excel/poi/test/Test2.java @@ -0,0 +1,49 @@ +package com.excel.poi.test; + +import com.excel.poi.ExcelBoot; +import com.excel.poi.entity.ErrorEntity; +import com.excel.poi.function.ImportFunction; + +import java.io.FileInputStream; +import java.io.FileNotFoundException; +import java.util.*; + +public class Test2 { + public static void main(String... args) throws FileNotFoundException { + + Map batchCounterMap = new HashMap(); + batchCounterMap.put("counter", 0); + batchCounterMap.put("bad", 0); + + final List batchCodes = new ArrayList(); + final StringBuffer error = new StringBuffer(); + String f = "C:\\TDDownload\\点价资源数据 (6) - 副本.xlsx"; +// String f = "C:\\TDDownload\\点价资源数据 (6) - 副本 - 副本.xlsx"; + ExcelBoot.ImportBuilder(new FileInputStream(f), ExcelVO.class, 2) + .importExcel(new ImportFunction() { + + /** + * @param sheetIndex 当前执行的Sheet的索引, 从1开始 + * @param rowIndex 当前执行的行数, 从1开始 + * @param inEntity Excel行数据的实体 + */ + @Override + public void onProcess(int sheetIndex, int rowIndex, ExcelVO inEntity) { + //将读取到Excel中每一行的inEntity数据进行自定义处理 + //如果该行数据发生问题,将不会走本方法,而会走onError方法 + batchCodes.add(inEntity.getCol_1()); + } + + /** + * @param errorEntity 错误信息实体 + */ + @Override + public void onError(ErrorEntity errorEntity) { + //将每条数据非空和正则校验后的错误信息errorEntity进行自定义处理 + error.append(errorEntity.getRowIndex()).append("行数据处理错误,").append(errorEntity.getErrorMessage()).append("\r\n"); + } + }, new ArrayList(Arrays.asList(0))); + System.out.println(batchCodes); + System.out.println(error); + } +} diff --git a/src/test/java/com/excel/poi/utils/ExcelUtilsTest.java b/src/test/java/com/excel/poi/utils/ExcelUtilsTest.java new file mode 100644 index 0000000..47befaa --- /dev/null +++ b/src/test/java/com/excel/poi/utils/ExcelUtilsTest.java @@ -0,0 +1,67 @@ +package com.excel.poi.utils; + +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.xssf.usermodel.XSSFWorkbook; + +import java.io.FileOutputStream; +import java.io.IOException; + +public class ExcelUtilsTest { + + public static void main(String... args) throws IOException { + writeColXlsx(); + + System.out.println(ExcelUtils.getExcelColumnIndex(-1, "XFD", -9)); + + } + + public static void writeColXlsx() throws IOException { + Workbook workbook = new XSSFWorkbook(); + Sheet sheet = workbook.createSheet("列"); + Row rowChar = sheet.createRow(0); + Row rowIndex = sheet.createRow(1); + Row rowActualIndex = sheet.createRow(2); + int colIndex = 0; + String tmp = null; + for(int i=0;i<26;i++) { + tmp = null; + tmp = String.valueOf((char) (i+65)); + rowChar.createCell(colIndex).setCellValue(tmp); + rowIndex.createCell(colIndex).setCellValue(ExcelUtils.getExcelColumnIndex(-1, tmp, -3)); + rowActualIndex.createCell(colIndex).setCellValue(colIndex); + colIndex++; + } + for(int i=0;i<26;i++) { + for(int j=0;j<26;j++) { + tmp = null; + tmp = String.valueOf((char) (i+65)) + String.valueOf((char) (j+65)); + rowChar.createCell(colIndex).setCellValue(tmp); + rowIndex.createCell(colIndex).setCellValue(ExcelUtils.getExcelColumnIndex(-1, tmp, -3)); + rowActualIndex.createCell(colIndex).setCellValue(colIndex); + colIndex++; + } + } + for(int i=0;i<26;i++) { + for(int j=0;j<26;j++) { + for(int k=0;k<26;k++) { + if(colIndex >= 16384) { + break; + } + tmp = null; + tmp = String.valueOf((char) (i+65)) + String.valueOf((char) (j+65)) + String.valueOf((char) (k+65)); + rowChar.createCell(colIndex).setCellValue(tmp); + rowIndex.createCell(colIndex).setCellValue(ExcelUtils.getExcelColumnIndex(-1, tmp, -3)); + rowActualIndex.createCell(colIndex).setCellValue(colIndex); + colIndex++; + } + } + } + + FileOutputStream fos = new FileOutputStream("d:\\col.xlsx"); + workbook.write(fos); + workbook.close(); + fos.close(); + } +} -- Gitee From 07a7f8468969fe04fd8812399630fb6806f39cce Mon Sep 17 00:00:00 2001 From: liujifei <327714@163.com> Date: Mon, 1 Jul 2019 15:41:43 +0800 Subject: [PATCH 2/2] =?UTF-8?q?fix=20bug=20=E8=AE=BE=E7=BD=AE=E5=88=97?= =?UTF-8?q?=E6=95=B0=E5=8F=B7=E6=97=A0=E6=95=88=E9=97=AE=E9=A2=98?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- src/main/java/com/excel/poi/excel/ExcelReader.java | 4 ++-- src/test/java/com/excel/poi/test/ExcelVO.java | 12 ++++++------ src/test/java/com/excel/poi/test/Test2.java | 7 ++++++- 3 files changed, 14 insertions(+), 9 deletions(-) diff --git a/src/main/java/com/excel/poi/excel/ExcelReader.java b/src/main/java/com/excel/poi/excel/ExcelReader.java index ac3856a..627b7bd 100644 --- a/src/main/java/com/excel/poi/excel/ExcelReader.java +++ b/src/main/java/com/excel/poi/excel/ExcelReader.java @@ -242,7 +242,7 @@ public class ExcelReader extends DefaultHandler { int index = Integer.parseInt(currentCellValue); currentCellValue = new XSSFRichTextString(mSharedStringsTable.getEntryAt(index)).toString(); } - if (!currentCellLocation.equals(previousCellLocation) && currentRowIndex != 0) { + if (!currentCellLocation.equals(previousCellLocation)) { for (int i = 0; i < countNullCell(currentCellLocation, previousCellLocation); i++) { cellsOnRow.put(excelCurrentCellIndex, ""); excelCurrentCellIndex++; @@ -251,8 +251,8 @@ public class ExcelReader extends DefaultHandler { if (!"".equals(currentCellValue.trim())) { String value = this.getCellValue(currentCellValue.trim()); cellsOnRow.put(excelCurrentCellIndex, value); - excelCurrentCellIndex++; } + excelCurrentCellIndex++; } else if (Constant.ROW.equals(name)) { if (null != endCellLocation) { diff --git a/src/test/java/com/excel/poi/test/ExcelVO.java b/src/test/java/com/excel/poi/test/ExcelVO.java index 01da8a0..3810630 100644 --- a/src/test/java/com/excel/poi/test/ExcelVO.java +++ b/src/test/java/com/excel/poi/test/ExcelVO.java @@ -4,17 +4,17 @@ import com.excel.poi.annotation.ImportField; public class ExcelVO { - @ImportField(regex = "65|65\\d{9}", regexMessage="非批号" , required = true) + @ImportField(regex = "65|65\\d{9}", regexMessage="非批号" , required = true, excelColumnIndex = 0) private String col_1; - @ImportField(regex = "[\\s\\S]*", regexMessage="列2校验失败", required = true) + @ImportField(regex = "[\\s\\S]*", regexMessage="列2校验失败", required = true, excelColumnIndex = 2) private String col_2; - @ImportField(regex = "((CF|cf|Cf|cF)\\d{4})|(\\u4e00\\u53e3\\u4ef7)", regexMessage="列3校验失败", required = true) + @ImportField(regex = "((CF|cf|Cf|cF)\\d{4})|(\\u4e00\\u53e3\\u4ef7)", regexMessage="列3校验失败", required = true, excelColumnIndex = 4) private String col_3; - @ImportField(regex = "\\d+(\\.\\d{1,2})?", regexMessage="金额格式不正确", required = true) + @ImportField(regex = "\\d+(\\.\\d{1,2})?", regexMessage="金额格式不正确", required = true, excelColumnIndex = 5) private String col_4; - @ImportField(regex = "[\\s\\S]*", regexMessage="列5校验失败") + @ImportField(regex = "[\\s\\S]*", regexMessage="列5校验失败", excelColumnIndex = 6) private String col_5; - @ImportField(regex = "[\\s\\S]*", regexMessage="列6校验失败") + @ImportField(regex = "[\\s\\S]*", regexMessage="列6校验失败", excelColumnIndex = 7) private String col_6; public String getCol_1() { diff --git a/src/test/java/com/excel/poi/test/Test2.java b/src/test/java/com/excel/poi/test/Test2.java index 8227ae0..ad3bef4 100644 --- a/src/test/java/com/excel/poi/test/Test2.java +++ b/src/test/java/com/excel/poi/test/Test2.java @@ -31,7 +31,12 @@ public class Test2 { public void onProcess(int sheetIndex, int rowIndex, ExcelVO inEntity) { //将读取到Excel中每一行的inEntity数据进行自定义处理 //如果该行数据发生问题,将不会走本方法,而会走onError方法 - batchCodes.add(inEntity.getCol_1()); + batchCodes.add(new StringBuffer(inEntity.getCol_1()).append("||") + .append(inEntity.getCol_2()).append("||") + .append(inEntity.getCol_3()).append("||") + .append(inEntity.getCol_4()).append("||") + .append(inEntity.getCol_5()).append("||") + .append(inEntity.getCol_6()).toString()); } /** -- Gitee