前言:

最近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. laravel5 数据库连接问题

    [PDOException] SQLSTATE[28000] [1045] Access denied for user ‘homestead’@’localhost’ (using password ...

  2. coding菜鸟养成记

    http://www.cnblogs.com/xdp-gacl/category/563690.html http://www.cnblogs.com/vincent-blog/p/4402327.h ...

  3. 算法系列001---dfs|多叉|解空间树理解

    1.研究范围 1)多叉树,图的遍历 2)回溯法的解空间树=多叉树的遍历 2.研究方法 我们现在研究的是多叉树的遍历,突然想到为什么不能直接用二叉树的遍历方法呢?我们抱着这个问题,先找到多叉树的结构不同 ...

  4. Zynq和microblaze的区别

    Zynq钩中PS端的外设之后不需要初始化过程,但是如果在microblaze中连接外设之后需要有初始化过程.

  5. 微信小程序之----navigator页面跳转

    navigator navigator跳转页面样式分为两种一种是左上角带返回按钮跳转到新的页面,另一种不带即在本页跳转,通过控制redirect属性 .js <view> <navi ...

  6. Mysql和Oracle的一些语法区别

    作为一个有追求的程序猿,当然要不断的学习,巴拉巴拉巴拉...好了,贴一个网址给大家,哈哈 MySQL与Oracle 差异比较:http://www.cnblogs.com/HondaHsu/p/364 ...

  7. 关于ThreadLocal和一般的线程同步的详细解释

    http://blog.csdn.net/lufeng20/article/details/24314381

  8. iOS 之 获取View所在控制器

    1. UIResponder UIViewController *uvc; UIResponder* nextResponder = [self.superview.superview.supervi ...

  9. YII 1.0 验证码

    public function actions(){ return array ( ‘captcha’=> array( ‘class’=> ‘CCatpchaAction’, ‘heig ...

  10. 使用(Drawable)资源———ShapeDrawable资源

    ShapeDrawable用于定义一个基本的几何图形(如矩形.圆形.线条等),定义ShapeDrawable的XML文件的根元素是<shape.../>元素,该元素可指定如下属性. and ...