前言:

最近PM来了一个需求,简单来说就是在录入数据时一条一条插入到系统显得非常麻烦,让我实现一个直接通过excel导入的方法一次性录入所有数据。网上关于excel导入导出的例子很多,但大多相互借鉴。经过思考,认为一百个客户在录入excel的时候,就会有一百个格式版本,所以在实现这个功能之前,所以要统一excel的格式。于是提供了一个通用excel模版的下载功能。当所有客户用模版录入好数据再上传到系统,后端对excel进行解析,然后再持久化到数据库。

概述:

   此工具类的几大特点

   1、基本导入导出

   2、提供excel模版自动生成及下载功能

   3、创建模版过程简单通用,只需要在实体类上进行注解

   4、springMVC框架

   5、模版可以生成下拉框选择列

废话不多说,上代码......

一、引入poi相关依赖(及spring上传文件相关配置,不再解释)

 <!-- apache poi start -->
<poi.version>3.14</poi.version>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>${poi.version}</version>
</dependency> <dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-scratchpad</artifactId>
<version>${poi.version}</version>
</dependency> <dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>${poi.version}</version>
</dependency>
<!-- apache poi end -->

二、excel导入导出工具类的实现

 /**
* @Description
* @author zhaomin E-mail:min.zhao@mljr.com
* @date 创建时间:2017年2月14日 下午2:13:30
* @version 1.0
*/
public class ImportExcelUtil {
final static String notnullerror = "请填入第{0}行的{1},{2}不能为空";
final static String errormsg = "第{0}行的{1}数据导入错误"; /**
* 导入Excel
*
* @param clazz
* @param xls
* @return
* @throws Exception
*/
@SuppressWarnings("rawtypes")
public static List importExcel(Class<?> clazz, InputStream xls) throws Exception {
try {
// 取得Excel
HSSFWorkbook wb = new HSSFWorkbook(xls);
HSSFSheet sheet = wb.getSheetAt(0);
Field[] fields = clazz.getDeclaredFields();
List<Field> fieldList = new ArrayList<Field>(fields.length);
for (Field field : fields) {
if (field.isAnnotationPresent(ModelProp.class)) {
ModelProp modelProp = field.getAnnotation(ModelProp.class);
if (modelProp.colIndex() != -1) {
fieldList.add(field);
}
}
}
EmployeeDTO employee = new EmployeeDTO();
// 行循环
List<ImportModel> modelList = new ArrayList<ImportModel>(sheet.getPhysicalNumberOfRows() * 2);
for (int i = 2; i < sheet.getPhysicalNumberOfRows(); i++) {
// 数据模型
ImportModel model = (ImportModel) clazz.newInstance();
int nullCount = 0;
Exception nullError = null;
for (Field field : fieldList) {
ModelProp modelProp = field.getAnnotation(ModelProp.class);
HSSFCell cell = sheet.getRow(i).getCell(modelProp.colIndex());
try {
if (cell == null || cell.toString().length() == 0) {
nullCount++;
if (!modelProp.nullable()) {
nullError = new Exception(StringUtil.format(notnullerror,
new String[] { "" + (1 + i), modelProp.name(), modelProp.name() })); }
} else if (field.getType().equals(Date.class)) {
if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(), new Date(parseDate(parseString(cell))));
} else {
BeanUtils.setProperty(model, field.getName(),
new Date(cell.getDateCellValue().getTime())); }
} else if (field.getType().equals(Timestamp.class)) {
if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(),
new Timestamp(parseDate(parseString(cell))));
} else {
BeanUtils.setProperty(model, field.getName(),
new Timestamp(cell.getDateCellValue().getTime()));
} } else if (field.getType().equals(java.sql.Date.class)) {
if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(),
new java.sql.Date(parseDate(parseString(cell))));
} else {
BeanUtils.setProperty(model, field.getName(),
new java.sql.Date(cell.getDateCellValue().getTime()));
}
} else if (field.getType().equals(java.lang.Integer.class)) {
if (Cell.CELL_TYPE_NUMERIC == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(), (int) cell.getNumericCellValue());
} else if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(), Integer.parseInt(parseString(cell)));
}
} else if (field.getType().equals(java.math.BigDecimal.class)) {
if (Cell.CELL_TYPE_NUMERIC == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(),
new BigDecimal(cell.getNumericCellValue()));
} else if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(), new BigDecimal(parseString(cell)));
}
} else {
if (Cell.CELL_TYPE_NUMERIC == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(),
new BigDecimal(cell.getNumericCellValue()));
} else if (Cell.CELL_TYPE_STRING == cell.getCellType()) {
BeanUtils.setProperty(model, field.getName(), parseString(cell));
}
}
} catch (Exception e) {
e.printStackTrace();
throw new Exception(StringUtil.format(errormsg, new String[] { "" + (1 + i), modelProp.name() })
+ "," + e.getMessage());
}
}
if (nullCount == fieldList.size()) {
break;
}
if (nullError != null) {
throw nullError;
}
modelList.add(model);
}
return modelList; } finally {
xls.close();
}
} private final static int colsizeN = 630;
private final static int colsizeM = 1000; /**
* 下载Excel模版
*
* @param clazz
* @param map
* @param rowSize
* @return
*/
public static InputStream excelModelbyClass(Class<?> clazz, Map<Integer, String[]> map, Integer rowSize) {
try {
if (!clazz.isAnnotationPresent(ModelTitle.class)) {
throw new Exception("请在此类型中加上ModelTitle注解");
}
if (rowSize == null) {
rowSize = 1000;
}
HSSFWorkbook wb = new HSSFWorkbook();
HSSFSheet sheet = wb.createSheet();
/**
* 设置标题样式
*/
HSSFCellStyle titleStyle = wb.createCellStyle();
titleStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontHeight((short) 400);
titleStyle.setFont(font);
HSSFCell titleCell = sheet.createRow(0).createCell(0); // 创建第一行,并在该行创建单元格,设置内容,做为标题行
/**
* 获取标题
*/
ModelTitle modelTitle = clazz.getAnnotation(ModelTitle.class);
titleCell.setCellValue(new HSSFRichTextString(modelTitle.name()));
titleCell.setCellStyle(titleStyle); Field[] fields = clazz.getDeclaredFields();
HSSFRow headRow = sheet.createRow(1);
int colSzie = 0;
/**
* 设置表头样式
*/
HSSFCellStyle headStyle = wb.createCellStyle();
headStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
HSSFFont headFont = wb.createFont();
headFont.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
headFont.setFontHeight((short) 240);
headStyle.setFont(headFont);
List<Integer> cells = new ArrayList<Integer>(); for (Field field : fields) {
if (field.isAnnotationPresent(ModelProp.class)) {
ModelProp modelProp = field.getAnnotation(ModelProp.class);
if (modelProp.colIndex() == -1)
continue;
cells.add(modelProp.colIndex());
HSSFCell cell = headRow.createCell(modelProp.colIndex());
cell.setCellValue(new HSSFRichTextString(modelProp.name()));
cell.setCellStyle(headStyle);
colSzie++;
sheet.autoSizeColumn((short) modelProp.colIndex());
sheet.setColumnWidth(modelProp.colIndex(), modelProp.name().length() * colsizeN + colsizeM); // 设置列为下拉框格式
if (map != null && map.get(new Integer(modelProp.colIndex())) != null) {
DVConstraint constraint = DVConstraint
.createExplicitListConstraint(map.get(modelProp.colIndex()));
CellRangeAddressList regions = new CellRangeAddressList(2, rowSize, modelProp.colIndex(),
modelProp.colIndex());
HSSFDataValidation dataValidation = new HSSFDataValidation(regions, constraint);
sheet.addValidationData(dataValidation);
}
}
}
HSSFCellStyle cellStyle = wb.createCellStyle();
HSSFDataFormat format = wb.createDataFormat();
cellStyle.setDataFormat(format.getFormat("@"));
for (int i = 2; i < rowSize; i++) {
HSSFRow row = sheet.createRow(i);
for (Integer integer : cells) {
HSSFCell cell = row.createCell(integer);
cell.setCellStyle(cellStyle);
}
}
sheet.addMergedRegion(new CellRangeAddress(0, 0, 0, colSzie - 1));
if (map != null) {
for (Integer colIndex : map.keySet()) {
DVConstraint constraint = DVConstraint.createExplicitListConstraint(map.get(colIndex));
CellRangeAddressList regions = new CellRangeAddressList(2, 1000, colIndex, colIndex);
HSSFDataValidation dataValidation = new HSSFDataValidation(regions, constraint);
sheet.addValidationData(dataValidation);
}
} ByteArrayOutputStream os = new ByteArrayOutputStream();
try {
wb.write(os);
} catch (IOException e) {
e.printStackTrace();
} byte[] b = os.toByteArray(); ByteArrayInputStream in = new ByteArrayInputStream(b);
return in;
} catch (Exception e) {
e.printStackTrace();
return null;
}
} private static String parseString(HSSFCell cell) {
return String.valueOf(cell).trim();
} private static long parseDate(String dateString) throws ParseException {
if (dateString.indexOf("/") == 4) {
return new SimpleDateFormat("yyyy/MM/dd").parse(dateString).getTime();
} else if (dateString.indexOf("-") == 4) {
return new SimpleDateFormat("yyyy-MM-dd").parse(dateString).getTime();
} else if (dateString.indexOf("年") == 4) {
return new SimpleDateFormat("yyyy年MM月dd").parse(dateString).getTime();
} else if (dateString.length() == 8) {
return new SimpleDateFormat("yyyyMMdd").parse(dateString).getTime();
} else {
return new Date().getTime();
}
} }

三、自定义spring注解

 @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface ModelProp{
public String name();
public int colIndex() default -1;
public boolean nullable() default true;
public String interfaceXmlName() default "";
}
 @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface ModelTitle{
public String name();
}

四、定义实体类父类

 public class ImportModel {

 }

五、定义实体类

 @ModelTitle(name="人员列表")
public class EmployeeDTO extends ImportModel implements Serializable { private static final long serialVersionUID = -3434719712955859295L; private Long id;
@ModelProp(name = "电话", colIndex = 1, nullable = false)
private String telephone; @ModelProp(name = "名称", colIndex = 0, nullable = false)
private String name; @ModelProp(name = "性别", colIndex = 2, nullable = false)
private Integer sex;
}

六、定义controller

 @RestController
@RequestMapping("/api/excelOpera")
public class ImportEmployeeController extends BaseController { private static Logger logger = LoggerFactory.getLogger(ImportEmployeeController.class);
/**
* 导入excel表
* @version 1.0
* @since 1.0
*/
@RequestMapping(path = "/importEmployee", method = RequestMethod.POST)
public RespMsg uploadExcel(@RequestParam("file") MultipartFile file, HttpServletRequest request) {
try{
  
List<EmployeeDTO> employeeDTOList = ImportExcelUtil.importExcel(EmployeeDTO.class, file.getInputStream());
//可做持久化操作,现只打印观察
for(EmployeeDTO employeeDTO : employeeDTOList){
logger.info("name=" + employeeDTO.getName() + ",telephone=" + employeeDTO.getTelephone()+",sex=" + employeeDTO.getSex());
}
}catch(Exception e){
logger.error(e.getMessage());
}
return null;
}
/**
* 导出excel模版
* @version 1.0
* @since 1.0
*/
@RequestMapping(path = "/downloadEmployeeModel", method = RequestMethod.GET)
public RespMsg downloadEmployeeModel(HttpServletResponse response) {
try{
response.setContentType("application/xls");
response.addHeader("Content-Disposition", "attachment;filename="+new String(("eeelist").getBytes("UTF-8"),"iso-8859-1")+".xls");
Map<Integer,String[]> paramMap = new HashMap<Integer,String[]>();
//excel第三行为下拉选择框
paramMap.put(2, new String[]{"man","women"});
BufferedInputStream input = new BufferedInputStream(ImportExcelUtil.excelModelbyClass(EmployeeDTO.class, paramMap, null));
byte buffBytes[] = new byte[1024];
OutputStream os = response.getOutputStream();
int read = 0;
while ((read = input.read(buffBytes)) != -1) {
os.write(buffBytes, 0, read);
}
os.flush();
os.close();
input.close();
return success("下载成功!");
}catch(Exception e){
logger.error("downloadEmployeeModel() catch Exception ",e);
return fail("下载失败!");
}
} }

至此全部工具类的实现已经完成,可以请求访问检验一下结果。

  下载下来的excel模版

  

  填写数据上传

  

  后台控制台打印输出结果

  

  检验结果能够达到预期效果且能通用,是不是很简单呢,欢迎大神们提出意见,小女子感谢了。

一个基于POI的通用excel导入导出工具类的简单实现及使用方法的更多相关文章

  1. Java基础学习总结(49)——Excel导入导出工具类

    在项目的pom文件中引入 <dependency> <groupId>net.sourceforge.jexcelapi</groupId> <artifac ...

  2. 【原创】POI操作Excel导入导出工具类ExcelUtil

    关于本类线程安全性的解释: 多数工具方法不涉及共享变量问题,至于添加合并单元格方法addMergeArea,使用ThreadLocal变量存储合并数据,ThreadLocal内部借用Thread.Th ...

  3. java中excel导入\导出工具类

    1.导入工具 package com.linrain.jcs.test; import jxl.Cell; import jxl.Sheet; import jxl.Workbook; import ...

  4. java Excel导入导出工具类

    本文章,导入导出依赖提前定义好的模板 package com.shareworx.yjwy.utils; import java.io.File; import java.io.FileInputSt ...

  5. 通用Excel文件导出工具类

    1:Excel格式 2:ExcelUtil.java import java.io.ByteArrayOutputStream; import java.io.IOException; import ...

  6. java简易excel导入导出工具(封装POI)

    Octopus 如何导入excel 如何导出excel github项目地址 Octopus Octopus 是一个简单的java excel导入导出工具. 如何导入excel 下面是一个excel文 ...

  7. Octopus——excel导入导出工具

    Octopus Octopus是一个简易的Excel导入导出工具.目前主要就两个功能: 导入:将excel中一行数据转换为指定的java对象,并通过指定的正则表达式检查合法性. 导出:按照给定的xml ...

  8. Excel导入导出工具(简单、好用且轻量级的海量Excel文件导入导出解决方案.)

    Excel导入导出工具(简单.好用且轻量级的海量Excel文件导入导出解决方案.) 置顶 2019-09-07 16:47:10 $9420 阅读数 261更多 分类专栏: java   版权声明:本 ...

  9. Excel导入导出帮助类

    /// <summary>    /// Excel导入导出帮助类    /// 记得引入 NPOI    /// 下载地址   http://npoi.codeplex.com/rele ...

随机推荐

  1. 移位操作<< >> 小计

    最近看代码移位操作的使用比较多,通过查阅资料搞明白了之前一直不明白的一个问题 移位效果: 左移位<<: 相当于乘以  2的左移位的相应次方   例如: x << 3    =& ...

  2. 从CK+库提取标记点信息

    1.CK+动态表情库介绍 The Extended Cohn-Kanade Dataset(CK+) 下载地址 这个数据库是在 Cohn-Kanade Dataset 的基础上扩展来的,发布于2010 ...

  3. Recovering a WiredTiger collection from a corrupt MongoDB installation

    Reference: http://www.alexbevi.com/blog/2016/02/10/recovering-a-wiredtiger-collection-from-a-corrupt ...

  4. SQL数据库修复/数据库置疑修复

    SQL数据库修复的三大核心技术: 1.磁盘阵列分析重组技术: 2.数据库恢复与修复技术: 3.SCSI盘物理故障开盘技术. 至今已经成功恢复数百台服务器的SQL数据库,用户覆盖全国. 导致SQL数据库 ...

  5. vi的基本操作

    vi的基本操作 a) 进入vi 在系统提示符号输入vi及文件名称后,就进入vi全屏幕编辑画面: $ vi myfile 不过有一点要特别注意,就是您进入vi之后,是处于「命令行模式(command m ...

  6. cordova环境搭建

    首先,需要了解一下cordova 和 phonegap 的关系.phoneGap是原先的名字,Cordova是phoneGap被捐给apache之后 用的项目名,phoneGap的名字也被保留了. 1 ...

  7. UVa 727 - Equation

    题目大意:给一个中缀表达式,转换成后缀表达式. 这类题一直不太会,让我想就是建一棵表达式树,然后后续遍历算了,可是建树的过程实在太麻烦了.今天才看到有中缀表达式转换成后缀表达式的算法,可以用栈进行实现 ...

  8. poi2007

    序:为什么写poi,zy说poi都是思路题目,不像hnoi妈的数据结构队..... 1.bzoj1102 题目大意:定义了一个山谷和山峰,求他们数量. 题解:这种题bfs咯,在bfs的时候记录一下相邻 ...

  9. Java中间件:淘宝网系统高性能利器(转)

    淘宝网是亚太最大的网络零售商圈,其知名度毋庸置疑,吸引着越来越多的消费者从街头移步这里,成为其忠实粉丝.如此多的用户和交易量,也意味着海量的信息处理,其背后的IT架构的稳定性.可靠性也显得尤为重要.那 ...

  10. 【WebStorm】前端工具开发利器webstrom专篇...更新中

    http://my.oschina.net/maomi/blog/137807#OSC_h2_5 WebStorm混搭svn WebStorm混搭nodeJS webstorm简单介绍 webstor ...