All Downloads are FREE. Search and download functionalities are using the official Maven repository.

gu.sql2java.excel.ExcelReader Maven / Gradle / Ivy

There is a newer version: 5.3.2
Show newest version
package gu.sql2java.excel;

import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static gu.sql2java.SimpleLog.log;
import static gu.sql2java.excel.BaseExcelReader.suffixOf;
import static net.gdface.bean.BeanPropertySupport.isEmpty;
import static com.google.common.base.Strings.nullToEmpty;
import static com.google.common.base.MoreObjects.firstNonNull;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.function.Consumer;

import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.web.multipart.MultipartFile;

import com.google.common.base.Predicates;
import com.google.common.collect.ContiguousSet;
import com.google.common.collect.DiscreteDomain;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.Range;
import com.google.common.primitives.Ints;

import gu.sql2java.excel.config.SheetConfig;

/**
 * EXCEL(.xls,.xlsx,.csv)文件导入,支持多工作薄导入
* * @author guyadong * @since 3.29.0 * */ public final class ExcelReader implements IExcelReader { @SuppressWarnings("rawtypes") private final ArrayList builders = Lists.newArrayList(); private Collection readerWorkSort; private Boolean debugOutput; /** * 默认构造方法,创建第一个工作薄配置 */ public ExcelReader() { this(null,new ArrayList<>(),null); } /** * 构造方法 * 创建第一个工作薄配置 * @param beanConsumer */ public ExcelReader(Consumer beanConsumer) { this(null,new ArrayList<>(),beanConsumer); } /** * 构造方法 * 创建第一个工作薄配置 * @param sheetConfig * @param beanConsumer */ public ExcelReader(SheetConfig sheetConfig, Consumer beanConsumer) { this(sheetConfig,new ArrayList<>(),beanConsumer); } /** * 构造方法 * 创建第一个工作薄配置 * @param sheetConfig * @param rows * @param beanConsumer */ public ExcelReader(SheetConfig sheetConfig, List rows, Consumer beanConsumer) { configSheet(sheetConfig, rows, beanConsumer,0); } public ExcelReader(SheetConfig sheetConfig) { this(sheetConfig,new ArrayList<>(),null); } public ExcelReader (BaseExcelReader builder) { configSheet(builder,0); } /** * 设置调试输出标志 * @param debugOutput */ public ExcelReader debugOutput(Boolean debugOutput) { this.debugOutput = debugOutput; return this; } /** * 打开/关闭嵌套字段优先模式,默认开启 * @param subFirst 为{@code true} 嵌套字段优先,否则按输入字段顺序处理 * @param indexes 工作薄解析配置对象索引 * @return 当前对象 * @since 3.29.9 */ public ExcelReader subFirst(boolean subFirst,int...indexes) { if(indexes != null) { for(int index:indexes) { if(index >= 0 && index < builders.size()) { getBuilder(index).subFirst(subFirst); } } } return this; } /** * 打开/关闭嵌套字段优先模式,默认开启 * @param subFirst 为{@code true} 嵌套字段优先,否则按输入字段顺序处理 * @param names 工作薄解析配置对象对应的工作薄名称 * @return 当前对象 * @since 3.29.9 */ public ExcelReader subFirst(boolean subFirst,String...names) { if(names != null) { for(String name:names) { if(!isEmpty(name)) { BaseExcelReader builder = getBuilderOrNull(name); if(null != builder) { builder.subFirst(subFirst); } } } } return this; } /** * 多工作薄导入的情况下, * 指定工作薄解析配置的执行顺序,如果不指定则从0开始顺序执行 * @param indexes * @return 当前对象 * @since 3.29.7 */ public ExcelReader readerWorkSort(Iterable indexes) { if(null != indexes) { this.readerWorkSort = Lists.newArrayList(Iterables.filter(indexes, Predicates.notNull())) ; } return this; } /** * 多工作薄导入的情况下, * 指定工作薄解析配置的执行顺序,如果不指定则从0开始顺序执行 * @param indexes * @return 当前对象 * @since 3.29.7 */ public ExcelReader readerWorkSort(int... indexes) { if(null != indexes) { this.readerWorkSort = Ints.asList(indexes); } return this; } /** * 顺序设置工作薄解析配置对象 * @param readers * @return 当前对象 * @since 3.29.8 */ public ExcelReader configSheet(Iterable> readers) { if(null != readers) { builders.clear(); builders.addAll(readers instanceof Collection ? (Collection>)readers : Lists.newArrayList(readers)); } return this; } /** * 顺序设置工作薄解析配置对象 * @param readers * @return 当前对象 * @since 3.29.8 */ public ExcelReader configSheet(BaseExcelReader... readers) { if(null != readers) { configSheet(Arrays.asList(readers)); } return this; } /** * 为指定索引设置工作薄解析配置 * @param reader * @param index 指定配置索引位置(0-based) * @return 当前对象 * @since 3.29.7 */ public >ExcelReader configSheet(B reader, int index) { checkArgument(index>=0,"INVALID index %s",index); for(int i = builders.size();i<=index;++i) { builders.add(null); } builders.set(index, checkNotNull(reader,"builder is null")); return this; } /** * 为指定索引设置工作薄解析配置 * @param sheetConfig * @param rows 工作薄的解析记录保存列表 * @param beanConsumer * @param index 指定配置索引位置(0-based) * @return 当前对象 */ @SuppressWarnings({ "unchecked", "rawtypes" }) public ExcelReader configSheet(SheetConfig sheetConfig, List rows, Consumer beanConsumer, int index) { BaseExcelReader builder = new BaseExcelReader(sheetConfig, rows, beanConsumer); configSheet(builder,index); return this; } /** * 为指定索引设置工作薄解析配置 * @param sheetConfig * @param beanConsumer * @param index 0-based index * @return 当前对象 * @see #configSheet(SheetConfig, List, Consumer, int) */ public ExcelReader configSheet(SheetConfig sheetConfig,Consumer beanConsumer, int index) { configSheet(sheetConfig, new ArrayList<>(), beanConsumer, index); return this; } /** * 为指定索引设置工作薄解析配置 * @param sheetConfig * @param index 0-based index * @return 当前对象 * @see #configSheet(SheetConfig, List, Consumer, int) */ public ExcelReader configSheet(SheetConfig sheetConfig, int index) { return configSheet(sheetConfig, new ArrayList<>(),null, index); } /** * 为指定索引设置工作薄解析配置 * @param beanConsumer * @param index 0-based index * @return 当前对象 * @see #configSheet(SheetConfig, List, Consumer, int) */ public ExcelReader configSheet(Consumer beanConsumer, int index) { return configSheet(new SheetConfig(), new ArrayList<>(),beanConsumer, index); } /** * 从{@link InputStream}将导入数据记录
* 多工作薄导入时,如果解析配置{@link SheetConfig} 指定了工作薄名称(参见 {@code SheetConfig.sheetName }字段 ),优先使用SheetName查找工作薄 * 如果没有指定工作薄名,则查找对应的索引获取 Sheet读取数据 * @param inputStream * @param format 文件格式,即文件后缀: .xls,xlxs,.cxv * @throws IOException */ @SuppressWarnings("rawtypes") public void read(InputStream inputStream, Charset charset, String format) throws IOException { switch(nullToEmpty(format).toLowerCase()) { case ".csv": new CSVReader(checkNotNull(builders.get(0),"Sheet Config(0) is null")) .debugOutput(debugOutput) .read(inputStream, charset, format); break; case ".xls": case ".xlsx":{ try(Workbook workbook = createWorkbook(inputStream, format)){ if(readerWorkSort == null) { /** 如果没有指定工作薄解析配置的执行顺序则从0开始顺序执行 */ readerWorkSort = ContiguousSet.create(Range.closedOpen(0, builders.size()), DiscreteDomain.integers()); } for(Integer i:readerWorkSort) { checkArgument(i>=0&&i=0 AND < %s required",i,builders.size()); BaseExcelReader builder = builders.get(i); if(null == builder) { continue; } SheetReader sheetReader = builder instanceof SheetReader ? (SheetReader)builder : new SheetReader(builder); sheetReader.debugOutput(debugOutput); Sheet sheet = null; // 如果指定了工作薄名称,优先使用SheetName查找工作薄 String sheetName = builder.getSheetConfig().getSheetName(); if(!"exportedExcel".equals(sheetName) && !isEmpty(sheetName)) { sheet = workbook.getSheet(sheetName); } if(null == sheet && i < workbook.getNumberOfSheets()) { sheet = workbook.getSheetAt(i); } checkState(null != sheet,"NOT FOUND Sheet for SheetConfig "+i); log("READ Sheet name '{}' index= {}",sheetName,i); sheetReader.read(sheet); } } break; } default: throw new IOException("UNSUPPORTED format :"+format); } } static Workbook createWorkbook(InputStream inputStream, String format) throws IOException { switch(nullToEmpty(format).toLowerCase()) { case ".xls": return new HSSFWorkbook(inputStream); case ".xlsx": return new XSSFWorkbook(inputStream); default: throw new IOException("UNSUPPORTED format :"+format); } } /** * 设置第一个工作薄解析配置对象的解析记录保存列表为{@code null}忽略 */ @Override public void setRows(List rows) { setRows(rows, 0); } /** * 返回第一个工作薄解析配置对象的解析记录保存列表 */ @Override public List getRows() { return getRows(0); } /** * 设置索引指定的工作薄的解析记录保存列表为{@code null}忽略, * 索引无效抛出异常 * @param * @param rows * @param index * @return 当前对象 */ public ExcelReader setRows(List rows,int index) { getBuilder(index).setRows(rows); return this; } /** * 设置工作薄名指定的工作薄的解析记录保存列表为{@code null}忽略, * 按工作薄名无法找到工作薄则抛出异常 * @param * @param rows * @param sheetName 工作薄名 * @return 当前对象 */ public ExcelReader setRows(List rows,String sheetName) { getBuilder(sheetName).setRows(rows); return this; } /** * 返回索引指定的工作薄的解析记录保存列表, * 索引无效抛出异常 * @param * @param index */ public List getRows(int index) { return getBuilder(index).getRows(); } /** * 返回工作薄名指定的工作薄的解析记录保存列表, * 按工作薄名无法找到工作薄则抛出异常 * @param * @param sheetName */ public List getRows(String sheetName) { return getBuilder(sheetName).getRows(); } private BaseExcelReader getBuilder(int index) { try { return checkNotNull(this.builders.get(index),"NOT DEFINE Sheet Config at index "+index); } catch (IndexOutOfBoundsException e) { throw new IllegalArgumentException("INVALID index "+ index); } } private BaseExcelReader getBuilder(String sheetName) { try { return Iterables.tryFind(builders, b->null != b && b.sheetConfig.getSheetName().equals(sheetName)).get(); } catch (IllegalStateException e) { throw new IllegalArgumentException("INVALID sheetName "+ sheetName); } } private BaseExcelReader getBuilderOrNull(String sheetName) { return Iterables.tryFind(builders, b->null != b && b.sheetConfig.getSheetName().equals(sheetName)).orNull(); } @Override public void setSheetConfig(SheetConfig sheetConfig) { this.builders.get(0).setSheetConfig(sheetConfig); } @Override public SheetConfig getSheetConfig() { return this.builders.get(0).getSheetConfig(); } @Override public void read(MultipartFile uplodFile, Charset charset) throws IOException { try(InputStream inputStream = checkNotNull(uplodFile,"uplodFile is null").getInputStream()){ log(firstNonNull(debugOutput,false),"UPLOAD FILE: {}",uplodFile.getOriginalFilename()); read(inputStream, charset, suffixOf(uplodFile.getOriginalFilename())); } } @Override public void read(String file, Charset charset) throws IOException { try(FileInputStream inputStream = new FileInputStream(checkNotNull(file,"file is null"))) { log(firstNonNull(debugOutput,false),"FILE: {}",file); read(inputStream, charset, suffixOf(file)); } } @Override public void read(File file, Charset charset) throws IOException { try(FileInputStream inputStream = new FileInputStream(checkNotNull(file,"file is null"))) { log(firstNonNull(debugOutput,false),"FILE: {}",file); read(inputStream, charset, suffixOf(file.getName())); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy