使用 Apache poi 导入Excel
本文主要记录Excel导入及模板下载,遇到的问题及注意事项。
第一节:Excel导入
/**
* @param returnMap
* @Title: dealExcelData
* @Description: TODO(保存Excel中的数据,并过滤重复的记录)
* @author: yanghai * @param: @param contents 存储 Excel中的内容
* @param: @param item 上传的Excel元素
* @param: @param request
* @param: @param repeatCount
* @return: void
* @throws
*/
private void dealExcelData(List<CompanyInvoiceRecord> contents, MultipartFile item, HttpServletRequest request, Integer repeatCount, Map<String, Object> returnMap) throws Exception
{
List<Integer> l = new ArrayList<Integer>();
Integer count = 0;
CompanyInvoiceRecord dto = null; //临时文件名称
String tempDir = "/files-" +DateTool.formatDate(System.currentTimeMillis(), "yyyy-MM-dd-HH-mm");
//临时文件全路径
String tempFileDir = request.getSession().getServletContext().getRealPath(tempDir);
//创建临时文件
File tempFile = new File(tempFileDir);
if(!tempFile.exists())
{
tempFile.mkdir();
} //获取原始文件全名称
String originalFilename = item.getOriginalFilename();
// 获取文件后缀
String suffix = "";
try
{
suffix = originalFilename.substring(originalFilename.lastIndexOf("."));
}
catch (Exception e)
{
e.printStackTrace();
throw new Exception("没有文件信息!");
}
//完整的文件目录
String fileName = tempFileDir + File.separator + originalFilename;
File newFile = new File(fileName); try
{
// 保存到一个目标文件中。
item.transferTo(newFile);
}
catch (Exception e)
{
e.printStackTrace();
throw new Exception("保存上传Excel文件失败!");
}
Workbook wb = null; FormulaEvaluator formulaEvaluator = null; try
{
FileInputStream inputStream = new FileInputStream(newFile); if(suffix.endsWith("xls"))
{
wb = new HSSFWorkbook(inputStream); formulaEvaluator = new HSSFFormulaEvaluator((HSSFWorkbook) wb);
}
else
{
wb = new XSSFWorkbook(inputStream); formulaEvaluator = new XSSFFormulaEvaluator((XSSFWorkbook) wb);
}
}
catch (IOException e)
{
// 删除目录
deleteDir(new File(tempFileDir));
e.printStackTrace(); } // 保存有效的 Excel行
List<Row> rowList = new ArrayList<Row>(); Sheet sheet = wb.getSheetAt(0);
if(null == sheet)
{
deleteDir(new File(tempFileDir));
throw new Exception("导入失败:导入文件中不存在sheet页!");
}
else
{
/*int lastRowNum = sheet.getLastRowNum();
System.out.println("==============="+lastRowNum);
if(lastRowNum > 5001)
{
throw new Exception("超过导入上限。最多导入5000条!");
}*/
try
{
for(Row row : sheet)
{
// 校验表头
if(row.getRowNum() == 0)
{
if(StringUtils.isNotEmpty(row.getCell(0).toString().trim()) && "付款日期".equals(row.getCell(0).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(1).toString().trim()) && "发票号".equals(row.getCell(1).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(2).toString().trim()) && "金额".equals(row.getCell(2).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(3).toString().trim()) && "税额".equals(row.getCell(3).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(4).toString().trim()) && "合计".equals(row.getCell(4).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(5).toString().trim()) && "公司名称".equals(row.getCell(5).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(6).toString().trim()) && "货物名称".equals(row.getCell(6).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(7).toString().trim()) && "申请人".equals(row.getCell(7).toString().trim())
&& StringUtils.isNotEmpty(row.getCell(8).toString().trim()) && "申请金额".equals(row.getCell(8).toString().trim())
)
{
continue;
}
else
{
deleteDir(new File(tempFileDir));
throw new Exception("表头信息错误!");
}
}
else if(row.getRowNum() >= 1)
{ try
{
if((null == row.getCell(0) || String.valueOf(row.getCell(0)).equals(""))
&& (null == row.getCell(1) || String.valueOf(row.getCell(1)).equals(""))
&& (null == row.getCell(2) || String.valueOf(row.getCell(2)).equals(""))
&& (null == row.getCell(3) || String.valueOf(row.getCell(3)).equals(""))
&& (null == row.getCell(4) || String.valueOf(row.getCell(4)).equals(""))
&& (null == row.getCell(5) || String.valueOf(row.getCell(5)).equals(""))
&& (null == row.getCell(6) || String.valueOf(row.getCell(6)).equals(""))
&& (null == row.getCell(7) || String.valueOf(row.getCell(7)).equals(""))
&& (null == row.getCell(8) || String.valueOf(row.getCell(8)).equals("")))
{
System.out.println("===公司费用 发票 导入 记录 导入===此行"+row.getRowNum()+"为空");
}
else
{
rowList.add(row);
}
} catch (Exception e1)
{
e1.printStackTrace();
}
}
} if(null != rowList && rowList.size() > 0)
{
int lastRowNum = rowList.size();
System.out.println("==============="+lastRowNum);
if(lastRowNum > 5001)
{
throw new Exception("超过导入上限。最多导入5000条!");
} for(Row row : rowList)
{
try
{
dto = new CompanyInvoiceRecord(); short lastCellNum = row.getLastCellNum();
if(lastCellNum < 1)
{
deleteDir(new File(tempFileDir));
throw new Exception("第" + row.getRowNum() + "行列数不足!");
} if(StringUtils.isNotEmpty(row.getCell(0).toString()))
{
// 付款日期
dto.setPayDate(readCellToStringToTrim(row.getCell(0)).toString().trim());
} row.getCell(1).setCellType(Cell.CELL_TYPE_STRING);
if(StringUtils.isNotEmpty(row.getCell(1).toString().trim()))
{
// 发票号码
dto.setInvoiceNum(readCell(row.getCell(1)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(2).toString()))
{
// 金额
dto.setAmount(readCell(row.getCell(2)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(3).toString()))
{
// 税额
dto.setTaxAmount(readCell(row.getCell(3)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(4).toString()))
{
// 合计
dto.setTotalAmount(readCell(row.getCell(4)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(5).toString()))
{
// 抬头
dto.setTitle(readCell(row.getCell(5)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(6).toString()))
{
// 货物名称
dto.setGoodsName(readCell(row.getCell(6)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(7).toString()))
{
// 申请人
dto.setApplicantName(readCell(row.getCell(7)).toString().trim());
}
if(StringUtils.isNotEmpty(row.getCell(8).toString()))
{
// 申请金额
dto.setApplyAmount(readCell(row.getCell(8)).toString().trim());
} if(StringUtils.isEmpty(dto.getPayDate()) && StringUtils.isEmpty(dto.getInvoiceNum())
&& StringUtils.isEmpty(dto.getAmount()) && StringUtils.isEmpty(dto.getTitle())
&& StringUtils.isEmpty(dto.getTaxAmount()) && StringUtils.isEmpty(dto.getTotalAmount())
&& StringUtils.isEmpty(dto.getGoodsName()) && StringUtils.isEmpty(dto.getApplicantName())
&& StringUtils.isEmpty(dto.getApplyAmount()))
{ }
else
{
Boolean flag = true;
if(contents.contains(dto)) // 已包含
{
repeatCount++;
flag = false;
}
if(flag)
{
contents.add(dto);
}
}
} catch (Exception e)
{
l.add(row.getRowNum()+1);
count++;
System.out.println("==公司费用导入异常:"+e);
}
}
} returnMap.put("count", count);
returnMap.put("repeatCount", repeatCount); System.out.println("======异常条数:"+count+",发生异常的行数分别是:"+l.toString());
// 删除excel
deleteDir(new File(tempFileDir)); } catch (Exception e)
{
e.printStackTrace();
deleteDir(new File(tempFileDir));
throw new Exception("请按照要求填写Excel的内容!");
}
}
} /**
* 删除文件夹及文件夹下的内容
* @param dir
* @return
*/
private boolean deleteDir(File dir)
{
if (dir.isDirectory())
{
String[] children = dir.list();
// 递归删除目录中的子目录下
for (int i = 0; i < children.length; i++)
{
boolean success = deleteDir(new File(dir, children[i]));
if (!success)
{
return false;
}
}
}
// 目录此时为空,可以删除
return dir.delete();
}
/**
* Numeric Cell type (0)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_NUMERIC = 0; // 数字类型 /**
* String Cell type (1)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_STRING = 1; // 字符串类型 /**
* Formula Cell type (2)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_FORMULA = 2; // 公式类型 /**
* Blank Cell type (3)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_BLANK = 3; // 空白类型 /**
* Boolean Cell type (4)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_BOOLEAN = 4; // 布尔类型 /**
* Error Cell type (5)
* @see #setCellType(int)
* @see #getCellType()
*/
public final static int CELL_TYPE_ERROR = 5; // 错误类型
/**
* @description:读取Excel单元格数据
* @param cell excel单元格
* @return String
*/
private static String readCell(Cell cell)
{
String cell_value = ""; if (cell != null)
{
switch (cell.getCellType())
{
case Cell.CELL_TYPE_BOOLEAN:
// 得到Boolean对象的方法
if (cell.getBooleanCellValue())
{
cell_value = "TRUE";
} else
{
cell_value = "FALSE";
}
break;
case Cell.CELL_TYPE_NUMERIC:
// 先看是否是日期格式
if (DateUtil.isCellDateFormatted(cell))
{
// 读取日期格式
cell_value = DateUtils.formatDate(cell.getDateCellValue(), "yyyy-MM-dd");
} else
{
// 读取数字
cell_value = String.valueOf(cell.getNumericCellValue());
}
break;
case Cell.CELL_TYPE_FORMULA:
// 读取公式的值
cell_value = cell.getCellFormula();
break;
case Cell.CELL_TYPE_STRING:
// 读取String
cell_value = cell.getRichStringCellValue().getString();
break;
case Cell.CELL_TYPE_ERROR:
cell_value = cell.getErrorCellValue() + "";
break;
case HSSFCell.CELL_TYPE_BLANK:
cell_value = "";
break;
default:
cell_value = "";
}
}
return cell_value;
}
第二节:Excel模板下载
@RequestMapping("/downExcel")
public ModelAndView downBlack(HttpServletRequest request, HttpServletResponse response) throws IOException
{
String realPathName = "";
String tempPath = "";
String fileName = "";
BufferedInputStream bis = null;
BufferedOutputStream bos = null;
try {
tempPath = request.getSession().getServletContext().getRealPath("/") + "/download/";
fileName = "batchReceivedTicketTemplate.xlsx";
realPathName = tempPath + fileName;
long fileLength = new File(tempPath + fileName).length();
// 文件下载设置response
response.setContentType("text/html;charset=utf-8");
request.setCharacterEncoding("UTF-8");
response.setContentType("application/x-msdownload;");
// 火狐
if (request.getHeader("User-Agent").toLowerCase().indexOf("firefox") > 0)
{
response.setHeader("Content-disposition", "attachment; filename=" + new String("批量收票导入模板.xlsx".getBytes("utf-8"), "ISO8859-1"));
}
else
{
response.setHeader("Content-Disposition", "attachment;filename=" + new String("批量收票导入模板.xlsx".getBytes("gb2312"), "ISO8859-1"));
}
response.setHeader("Content-Length", String.valueOf(fileLength));
// 从模板获取输入流
bis = new BufferedInputStream(new FileInputStream(realPathName));
// 输出流
bos = new BufferedOutputStream(response.getOutputStream());
// 读取文件流输出
byte[] buff = new byte[2048];
int bytesRead;
while (-1 != (bytesRead = bis.read(buff, 0, buff.length)))
{
bos.write(buff, 0, bytesRead);
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
finally
{
if (bis != null)
bis.close();
if (bos != null)
bos.close();
}
return null;
}
这段代码,很清晰,在实现的过程中,要注意一点,就是模板的后缀要和下载到的模板文件后缀保持一致。否则在导入的时候会报下面这个异常信息提示:

使用 Apache poi 导入Excel的更多相关文章
- 项目一:第四天 1、快递员的条件分页查询-noSession,条件查询 2、快递员删除(逻辑删除) 3、基于Apache POI实现批量导入区域数据 a)Jquery OCUpload上传文件插件使用 b)Apache POI读取excel文件数据
1. 快递员的条件分页查询-noSession,条件查询 2. 快递员删除(逻辑删除) 3. 基于Apache POI实现批量导入区域数据 a) Jquery OCUpload上传文件插件使用 b) ...
- Java 使用poi导入excel,结合xml文件进行数据验证的例子(增加了jar包)
ava 使用poi导入excel,结合xml文件进行数据验证的例子(增加了jar包) 假设现在要做一个通用的导入方法: 要求: 1.xml的只定义数据库表中的column字段,字段类型,是否非空等条件 ...
- Java开发小技巧(六):使用Apache POI读取Excel
前言 在数据仓库中,ETL最基础的步骤就是从数据源抽取所需的数据,这里所说的数据源并非仅仅是指数据库,还包括excel.csv.xml等各种类型的数据接口文件,而这些文件中的数据不一定是结构化存储的, ...
- 利用Apache POI操作Excel
最近在做接口,有个功能是利用Excel导入汽车发动机所需零件信息到线上系统中.简单回顾一下之前学过的用java操作Excel. 1.maven配置Apache POI pom.xml中配置POIjar ...
- 在java poi导入Excel通用工具类示例详解
转: 在java poi导入Excel通用工具类示例详解 更新时间:2017年09月10日 14:21:36 作者:daochuwenziyao 我要评论 这篇文章主要给大家介绍了关于在j ...
- apache POI 导出excel相关方法
apache POI 操作excel无比强大.同时有操作word和ppt的接口. 下面讲解poi中常用方法. 1,设置列宽 HSSFSheet sheet = wb.getSheetAt(0); sh ...
- poi导入Excel,数字科学记数法转换
在这里分享一下使用poi 导入Excel时 把数字转换为科学记数法的解决方法: 就是使用DecimalFormat对 i 进行了格式化 结果为:
- 使用Apache POI导出Excel小结--导出XLS格式文档
使用Apache POI导出Excel小结 关于使用Apache POI导出Excel我大概会分三篇文章去写 使用Apache POI导出Excel小结--导出XLS格式文档 使用Apache POI ...
- Java使用Apache POI进行Excel导入和导出
Manve依赖 <!-- https://mvnrepository.com/artifact/org.apache.poi/poi-ooxml --> <dependency> ...
随机推荐
- mysql中把字符串转成时间戳进行对比
::") ");
- PHP之十六个魔术方法详细介绍
PHP中把以两个下划线__开头的方法称为魔术方法(Magic methods),这些方法在PHP中充当了举足轻重的作用.这里进行详细介绍,感兴趣的小伙伴们可以参考一下. PHP中把以两个下划线__开头 ...
- android电话状态的监听
电话状态的监听: /*** * 继承PhoneStateListener类,我们能够又一次其内部的各种监听方法 然后通过手机状态改变时,系统自己主动触发这些方法来实现我们想要的功能 */ class ...
- HTML_<select>
1.设置select只读不可编辑且select的值可传递 (1) <select onfocus="this.defaultIndex=this.selectedIndex;" ...
- UVA 10209
10209 - Is This Integration ? #include <stdio.h> #include <math.h> /* */ //多次错误都是因为我将PI定 ...
- 【cogs182】【USACO Jan07】均衡队形【st表】
题目描写叙述 农夫约翰的 N (1 ≤ N ≤ 50,000) 头奶牛,每天挤奶时总会按相同的顺序站好. 一日.农夫约翰决定为奶牛们举行一个"终极飞盘"比赛.为简化问题.他将从奶牛 ...
- wps文档怎样去除广告
安装完 WPS 之后右键—属性—打开文件夹位置(如图) 接下来进入 10.1.0.6929 文件夹内,再次进入 office6 文件夹内,找到 wpscenter 应用程序,将其删除.此应用包含定时弹 ...
- Java研发工程师面试题
基础题 一.String,StringBuffer, StringBuilder 的区别是什么?String为什么是不可变的?1. String是字符串常量,StringBuffer和StringBu ...
- 2016 acm香港网络赛 A题. A+B Problem (FFT)
原题地址:https://open.kattis.com/problems/aplusb FFT代码参考kuangbin的博客:http://www.cnblogs.com/kuangbin/arch ...
- iOS - 富文本
iOS--NSAttributedString超全属性详解及应用(富文本.图文混排) ios项目中经常需要显示一些带有特殊样式的文本,比如说带有下划线.删除线.斜体.空心字体.背景色.阴影以及图文 ...