一、使用poi
1.1测试导入导出文件
导出99乘法表
@Test
public void testPoiExport()throws Exception{
//创建一个工作薄对象 --easypoi
SXSSFWorkbook workbook = new SXSSFWorkbook();
//在工作薄里面创建表格
Sheet sheet = workbook.createSheet("99乘法表");
for(int i=1;i<=9;i++){
Row row = sheet.createRow(i - 1);
for(int j=1;j<=i;j++){
Cell cell = row.createCell(j - 1);
cell.setCellValue(i+"*"+j+"="+i*j);
}
}
//把内容写到99乘法里面
FileOutputStream fos = new FileOutputStream("99.xlsx");
workbook.write(fos);
fos.close();
}
导入文件
@Test
public void testPoiimport()throws Exception{
//构造一个文件出来
File file = new File("empread.xlsx");
//得到一个输入流
FileInputStream fis = new FileInputStream(file);
//得到一个工作薄
XSSFWorkbook workbook = new XSSFWorkbook(fis);
//第一个表格
XSSFSheet sheet = workbook.getSheetAt(0);
//拿到所有的行号
int lastRowNum = sheet.getLastRowNum(); //5
// System.out.println("lastRowNum:"+lastRowNum);
//0-5
for(int i=2;i<=lastRowNum;i++){
Row row = sheet.getRow(i);
short lastCellNum = row.getLastCellNum();
//System.out.println("lastCellNum:"+lastCellNum);
// 0,1,2,3 循环每一行数据
for(int j=0;j<lastCellNum;j++){
// 每行的单元格
Cell cell = row.getCell(j);
System.out.print(cell.getStringCellValue()+" ");
}
//换行 循环下一行数据
System.out.println();
}
}
二、 easypoi
2.1 easypoi是什么
easypoi 就是对poi进行封装,主打简单,do less do more ,就算不懂底层的poi的操作
,也可以操作导入导出,而且提供的功能在项目里够用
2.2 easypoi的特点
1.设计精巧,使用简单
2.接口丰富,扩展简单
3.默认值多,write less do more
4.spring mvc支持,web导出可以简单明了
2.3 使用easypoi(掌握)
(1)导入jar包 —加入pom.xml的依赖
<dependency>
<groupId>cn.afterturn</groupId>
<artifactId>easypoi-base</artifactId>
<version>3.2.0</version>
</dependency>
<dependency>
<groupId>cn.afterturn</groupId>
<artifactId>easypoi-web</artifactId>
<version>3.2.0</version>
</dependency>
<dependency>
<groupId>cn.afterturn</groupId>
<artifactId>easypoi-annotation</artifactId>
<version>3.2.0</version>
</dependency>
(2)准备一个类 比如EasyEmployee 和EasyDept
添加注解@Excel
导出相应的字段
处理boolean 可以处理图片类型 可以日期类型 可以处理关联对象
@ExcelTarget("emp")
public class EasyEmployee extends BaseDomain{
@Excel(name = "员工姓名")
private String name;
@Excel(name="邮件",width = 20)
private String email;
//性别
@Excel(name="性别",replace={"男_true","女_false"})
private boolean sex;
@Excel(name="年纪")
private Integer age;
//出生日期
@Excel(name="生日",format = "yyyy-MM-dd")
private Date birthday;
@ExcelEntity
private EasyDept dept;
@Excel(name="头像",type = 2)
private String headImage;
}
(3)调用工具方法 导入导出
//导入
List<EasyEmployee> easyEmployees = ExcelImportUtil.importExcel(file, EasyEmployee.class, importParams);
//导出方法
Workbook workbook = ExcelExportUtil.exportExcel(new ExportParams(), EasyDept.class,
list);
三、easypoi加入项目
3.1导出功能
(1)导入依赖jar包
(2)在页面添加导出功能
<form id="searchForm" action="/employee/download" method="post">
用户名: <input name="username" class="easyui-textbox" style="width:120px;height:32px">
邮件: <input name="email" class="easyui-textbox" style="width:80px;height:32px">
部门 :
<input class="easyui-combobox" name="departmentId"
data-options="valueField:'id',textField:'name',panelHeight:'auto',url:'/util/queryDepartment'" />
<%-- <input class="easyui-combobox" name="departmentId"
data-options="valueField:'id',textField:'name',panelHeight:'auto',url:'/util/queryDepartment'">--%>
<a href="#" data-method="search" class="easyui-linkbutton" iconCls="icon-search">查找</a>
<!-- 这部门是查询的功能 点击下面的代码 等效于提交表单-->
<button class="easyui-linkbutton" iconCls="icon-redo">导出</button>
</form>
(3)提交到EmployeeController 里面 download这个方法里面进行导出功能
(a)查询导出数据
List<Employee> employees = employeeService.findByQuery(employeeQuery);
(b)处理图片
String realPath = req.getServletContext().getRealPath("");
employees.forEach(e->{
//E:\JavaEEIdeacode\aisell\out\artifacts\aisell_Web_exploded\images\head
e.setHeadImage(realPath+e.getHeadImage());
System.out.println(e.getHeadImage());
});
©设置导出参数
ExportParams params = new ExportParams("员工数据", "员工表格1", ExcelType.XSSF);
//固定两列的意思
params.setFreezeCol(2);
map.put(NormalExcelConstants.DATA_LIST, employees); // 数据集合
map.put(NormalExcelConstants.CLASS, Employee.class);//导出实体
map.put(NormalExcelConstants.PARAMS, params);//参数
map.put(NormalExcelConstants.FILE_NAME, "xxx");//文件名称
(d)导出视图处理
<context:component-scan base-package="cn.afterturn.easypoi.view" />
<bean id="beanNameViewResolver" class="org.springframework.web.servlet.view.BeanNameViewResolver" p:order="0" />
(e)扫描视图的包
<context:component-scan base-package="cn.itsource.aisell.common"/>
导出代码的处理
public String download(EmployeeQuery employeeQuery, ModelMap map, HttpServletRequest req){
//根据查询 查询数据
List<Employee> employees = employeeService.findByQuery(employeeQuery);
String realPath = req.getServletContext().getRealPath("");//E:\JavaEEIdeacode\aisell\src\main\web/images/head/avatar1.jpg
employees.forEach(e->{
//E:\JavaEEIdeacode\aisell\out\artifacts\aisell_Web_exploded\images\head
e.setHeadImage(realPath+e.getHeadImage());
System.out.println(e.getHeadImage());
});
//导出参数设置
//E:\JavaEEIdeacode\aisell\src\main\web\images\head\sss.jpg
ExportParams params = new ExportParams("员工数据", "员工表格1", ExcelType.XSSF);
//固定两列的意思
params.setFreezeCol(2);
map.put(NormalExcelConstants.DATA_LIST, employees); // 数据集合
map.put(NormalExcelConstants.CLASS, Employee.class);//导出实体
map.put(NormalExcelConstants.PARAMS, params);//参数
map.put(NormalExcelConstants.FILE_NAME, "xxx");//文件名称
return NormalExcelConstants.EASYPOI_EXCEL_VIEW;//View名称
}
3.3 导入功能
(1)新增页面import.jsp页面
准备form表单,form表单里面提供一个上传的文件组件 easyui-filebox
<form action="/import/employeeXlsx" method="post" enctype="multipart/form-data">
<input class="easyui-filebox" name="empFile" style="width:80%"
data-options="prompt:'选择一个文件...',buttonText: '选择文件'" />
<button class="easyui-linkbutton">导入</button>
</form>
(2)点击导入 提交到后台进行导入处理
a)设置导入参数 --开启验证功能 开启自定义验证
写一个类 去实现 接口
@Component
public class EmployeeExcelVerifyHandler implements IExcelVerifyHandler<Employee> {
@Autowired
private IEmployeeService employeeService;
@Override
public ExcelVerifyHandlerResult verifyHandler(Employee employee) {
ExcelVerifyHandlerResult result = new ExcelVerifyHandlerResult(true);
//检查用户名是否重复的方法
boolean flag = employeeService.checkUsername(employee.getUsername());
//检查用户名如果重复
if(!flag){
result.setSuccess(false);
result.setMsg("用户名重复");
}
return result;
}
}
注意 扫描的包
<context:component-scan base-package="cn.itsource.aisell.common"/>
开启验证 – 是否为null 最大值
importParams.setNeedVerfiy(true);
//加入自定义验证类 --验证用户名是否重复
importParams.setVerifyHandler(employeeExcelVerifyHandler);
b)调用方法进行导入
ExcelImportResult<Employee> result = ExcelImportUtil.importExcelMore(empFile.getInputStream(),
Employee.class, importParams);
c)对于成功的结果 存入数据库
对于失败的结果 导出到前台展示
@RequestMapping("/employeeXlsx")
public String employeeXlsx(MultipartFile empFile, HttpServletResponse response) throws Exception {
ImportParams importParams = new ImportParams();
importParams.setTitleRows(1);
//开启验证 -- 是否为null 最大值
importParams.setNeedVerfiy(true);
//加入自定义验证类 --验证用户名是否重复
importParams.setVerifyHandler(employeeExcelVerifyHandler);
//完成导入 --调用导入
//导入并且验证功能
ExcelImportResult<Employee> result = ExcelImportUtil.importExcelMore(empFile.getInputStream(),
Employee.class, importParams);
result.getFailList().forEach(e->{
System.out.println("失败数据:"+e);
});
//导入的数据 保存数据库
result.getList().forEach(e->{
System.out.println(e);
//初始密码
e.setPassword("123456");
//根据部门名称 查询部门的id
Department dept = departmentService.findByName(e.getDepartment().getName());
e.setDepartment(dept);
//存入数据库
employeeService.save(e);
});
//把验证失败两个数据 导出到前台 error.xlsx
if(result.isVerfiyFail()){
//如果验证失败,代码到这里面来
//失败的文件已经准备好了
Workbook failWorkbook = result.getFailWorkbook();
//把这个文件导出
response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"); //mime类型
response.setHeader("Content-disposition", "attachment;filename=error.xlsx");
response.setHeader("Pragma", "No-cache");//设置不要缓存
OutputStream ouputStream = response.getOutputStream();
failWorkbook.write(ouputStream);
ouputStream.flush();
ouputStream.close();
}
return "import";
}
3.4 验证功能
自定义验证
@Component
public class EmployeeExcelVerifyHandler implements IExcelVerifyHandler<Employee> {
@Autowired
private IEmployeeService employeeService;
@Override
public ExcelVerifyHandlerResult verifyHandler(Employee employee) {
ExcelVerifyHandlerResult result = new ExcelVerifyHandlerResult(true);
//检查用户名是否重复的方法
boolean flag = employeeService.checkUsername(employee.getUsername());
//检查用户名如果重复
if(!flag){
result.setSuccess(false);
result.setMsg("用户名重复");
}
return result;
}
}