一、简介
1、Excel文件的组织形式和POI中基本结构:
- 一个Excel文件对应一个workbook
- 一个workerbook是有若干个sheet组成的
- 一个sheet有多个row
- 一个row一般存在多个cell
2、Excel和POI对象对应关系:
Excel结构 | POI对象 |
excel 的文档对象 | HSSFWorkbook |
excel 的表单(sheet) | HSSFSheet |
excel 的行 | HSSFRow |
excel 的格子单元 | HSSFCell |
3、POI创建Excel的步骤
- 生成文档对象HSSHWorkbook。
- 通过HSSFWorkbook生成表单HSSFSheet。
- 通过HSSFSheet生成行HSSFRow
- 通过HSSFRow生成单元格HSSFCell。
二、利用POI实现excel的导出
1、配置POM
<!-- https://mvnrepository.com/artifact/org.apache.poi/poi -->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>5.0.0</version>
</dependency>
2. 页面添加导出按钮
<el-button type="success" @click="downExcel">导出数据</el-button>
3. 添加处理函数
downExcel(){
window.location.href="http://localhost:9000/user/excel";
},
4. 工具类PageData,用来传递数据
package com.wnxy.wateraffair.util;
/**
* @author :fengSir
* @date :Created By 2022-08-30 10:07
* @description :TODO
*/
import java.util.HashMap;
import java.util.Map;
public class PageData extends HashMap {
Map map = null;
public PageData() {
map = new HashMap();
}
public String getString(Object key) {
return String.valueOf( map.get(key));
}
@SuppressWarnings("unchecked")
@Override
public Object put(Object key, Object value) {
return map.put(key, value);
}
@Override
public Object remove(Object key) {
return map.remove(key);
}
public int size() {
// TODO Auto-generated method stub
return map.size();
}
}
5. 导出Excel核心处理类ObjectExcelView,继承自AbstractXlsView
package com.wnxy.wateraffair.util;
import org.apache.poi.ss.usermodel.*;
import org.springframework.web.servlet.view.document.AbstractXlsView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.Map;
public class ObjectExcelView extends AbstractXlsView {
@Override
protected void buildExcelDocument(Map<String, Object> model, Workbook workbook, HttpServletRequest request, HttpServletResponse response) throws Exception {
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
String filename = sdf.format(new Date());
//设置下载头部文件信息
response.setContentType("application/octet-stream");
response.setHeader("Content-Disposition", "attachment;filename=" + filename + ".xls");
//创建sheet,相当于一个excelsheel
Sheet sheet = workbook.createSheet("sheet1");
//得到excel标题内容
List<String> titles = (List<String>) model.get("titles");
int len = titles.size();
//设置单元格样式
CellStyle cellStyle = workbook.createCellStyle(); //标题样式
cellStyle.setAlignment(HorizontalAlignment.CENTER);//水平居中
cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);//垂直居中
//设置字体
Font headerFont = workbook.createFont(); //标题字体
headerFont.setBold(true);
headerFont.setFontHeightInPoints((short) 11);
cellStyle.setFont(headerFont);
short width = 20, height = 25 * 20;
sheet.setDefaultColumnWidth(width);
//创建第一行,用来放标题
Row header = sheet.createRow(0);
for (int i = 0; i < len; i++) { //设置标题
String title = titles.get(i);
Cell cell = header.createCell(i);
cell.setCellValue(title);
cell.setCellStyle(cellStyle);
}
header.setHeight(height);
//设置内容样式
CellStyle contentStyle = workbook.createCellStyle(); //内容样式
contentStyle.setAlignment(HorizontalAlignment.CENTER);//水平居中
//得到controller传过来导出的数据,并填充到每一行中
List<PageData> varList = (List<PageData>) model.get("varList");
int varCount = varList.size();
for (int i = 0; i < varCount; i++) {
Row userRow = sheet.createRow(i + 1);
PageData vpd = varList.get(i);
int mapLen = vpd.size();
for (int j = 0; j < mapLen; j++) {
String varStr = vpd.getString("var" + (j + 1)) != null ? vpd.getString("var" + (j + 1)) : "";
Cell cell = userRow.createCell(j);
cell.setCellValue(varStr);
cell.setCellStyle(contentStyle);
}
}
}
}
6. Controller功能实现
@RequestMapping("excel")
public ModelAndView exportExcel(@RequestParam(value = "name", required = false) String name,
@RequestParam(value = "dateRange", required = false) String[] dateRange) {
String[] dateRangeArr = dateRange == null ? new String[]{} : dateRange;
UserDto user = new UserDto();
user.setDateRange(dateRangeArr);
user.setName(name);
ModelAndView mv = new ModelAndView();
PageData pd = new PageData();
try {
Map<String, Object> dataMap = new HashMap<>();
List<String> titles = new ArrayList<>();
titles.add("用户编号"); // 1
titles.add("姓名"); // 2
titles.add("电话");//3
titles.add("部门"); // 4
titles.add("上级"); // 5
titles.add("籍贯"); // 6
titles.add("生日"); // 7
dataMap.put("titles", titles);
List<UserVO> varOList = userService.queryAllExcel(user);
List<PageData> varList = new ArrayList<>();
for (int i = 0; i < varOList.size(); i++) {
PageData vpd = new PageData();
vpd.put("var1", varOList.get(i).getId()); // 1
vpd.put("var2", varOList.get(i).getUname()); // 2
vpd.put("var3", varOList.get(i).getTel()); // 3
vpd.put("var4", varOList.get(i).getDeptName()); // 4
vpd.put("var5", varOList.get(i).getUsuperior()); // 5
vpd.put("var6", varOList.get(i).getUorigo()); // 6
Date ubirthday = varOList.get(i).getUbirthday();
vpd.put("var7", DateUtil.convertDateToString(ubirthday)); // 7
varList.add(vpd);
}
dataMap.put("varList", varList);
ObjectExcelView erv = new ObjectExcelView();
mv = new ModelAndView(erv, dataMap);
} catch (Exception e) {
e.printStackTrace();
}
return mv;
}
7. Map文件
<select id="queryAllExcel" resultMap="UserVoResultMap">
select
su.*,sd.d_name
from user su left join dept sd on sd.d_id = su.d_id
<where>
<if test="name!=null and name!=''">
u_name like '%${name}%'
</if>
<if test="dateRange.length!=0">
and u_birthday between #{dateRange[0],javaType=string} and #{dateRange[1],javaType=string}
</if>
</where>
</select>
三、数据导入功能
1、页面
<el-col :span="4">
<el-upload class="clearfix" action="http://localhost:9000/user/importExcel" name="excel"
:on-progress="onExcelProgress" :on-success="handleExcelSuccess"
accept=".csv, application/vnd.openxmlformats-officedocument.spreadsheetml.sheet, application/vnd.ms-excel"
multiple :limit="1">
<el-button type="primary">数据导入</el-button>
</el-upload>
</el-col>
2、axios回调
handleExcelSuccess(res, result) {
this.$message.success("导入成功!!");
if (result.response.code == 200) {
this.loadUsers();
}
},
3、核心处理类
package com.wnxy.wateraffair.util;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import java.io.File;
import java.io.FileInputStream;
import java.util.ArrayList;
import java.util.List;
public class ObjectExcelRead {
/**
* @param filepath //文件路径
* @param filename //文件名
* @param startrow //开始行号
* @param startcol //开始列号
* @param sheetnum //sheet
* @return list
*/
public static List<Object> readExcel(String filepath, String filename, int startrow, int startcol, int sheetnum) {
List<Object> varList = new ArrayList<Object>();
try {
File target = new File(filepath, filename);
FileInputStream fi = new FileInputStream(target);
HSSFWorkbook wb = new HSSFWorkbook(fi);
HSSFSheet sheet = wb.getSheetAt(sheetnum); //sheet 从0开始
int rowNum = sheet.getLastRowNum() + 1; //取得最后一行的行号
for (int i = startrow; i < rowNum; i++) { //行循环开始
PageData varpd = new PageData();
HSSFRow row = sheet.getRow(i); //行
int cellNum = row.getLastCellNum(); //每行的最后一个单元格位置
for (int j = startcol; j < cellNum; j++) { //列循环开始
HSSFCell cell = row.getCell(Short.parseShort(j + ""));
Object cellValue = null;
if (null != cell) {
switch (cell.getCellType()) { // 判断excel单元格内容的格式,并对其进行转换,以便插入数据库
case _NONE:
cellValue = String.valueOf((int) cell.getNumericCellValue());
break;
case STRING:
cellValue = cell.getStringCellValue();
break;
case NUMERIC:
cellValue = cell.getNumericCellValue() + "";
// cellValue = String.valueOf(cell.getDateCellValue());
break;
case BLANK:
cellValue = "";
break;
case BOOLEAN:
cellValue = String.valueOf(cell.getBooleanCellValue());
break;
case ERROR:
cellValue = String.valueOf(cell.getErrorCellValue());
break;
}
} else {
cellValue = "";
}
varpd.put("var" + j, cellValue);
}
varList.add(varpd);
}
} catch (Exception e) {
e.printStackTrace();
}
return varList;
}
}
4、控制器
@RequestMapping("importExcel")
public Result readExcel(
@RequestParam(value="excel",required=false) MultipartFile file,HttpServletRequest request
) throws Exception{
System.out.println("----------------------------------------------");
PageData pd = new PageData();
if (null != file && !file.isEmpty()) {
String filePath = request.getSession().getServletContext().getRealPath("/upload"); //文件上传路径
String fileName = FileUpload.fileUp(file, filePath, "userexcel");//执行上传
List<PageData> listPd = (List) ObjectExcelRead.readExcel(filePath, fileName, 1, 0, 0); //执行读EXCEL操作,读出的数据导入List 2:从第2行开始;0:从第A列开始;0:第0个sheet
for(int i=0;i<listPd.size();i++){
User user = new User();
String strId = listPd.get(i).getString("var0");
int id=0;
if(strId.indexOf(".")!=-1){
id = Integer.parseInt(strId.substring(0,strId.indexOf(".")));
}else {
id = Integer.parseInt(strId);
}
user.setUId(id);
user.setUName(listPd.get(i).getString("var1"));
String var2 = listPd.get(i).getString("var2");
if(var2.indexOf(".")!=-1){
var2 = var2.substring(0,var2.indexOf("."));
}
user.setUTel(var2);
String strDeptName = listPd.get(i).getString("var3");
int deptId=0;
switch (strDeptName){
case "总务部":deptId = 1;break;
case "职能部":deptId = 2;break;
case "设备部":deptId = 3;break;
case "采购部":deptId = 4;break;
}
user.setDId(deptId);
user.setUSuperior(listPd.get(i).getString("var4"));
user.setUOrigo(listPd.get(i).getString("var5"));
user.setUBirthday(DateUtil.convertStringToDate(listPd.get(i).getString("var6")));
userService.insertSelective(user);
}
}
return Result.ok();
}
5、上传工具类
package com.wnxy.wateraffair.util;
import org.springframework.web.multipart.MultipartFile;
import java.io.File;
import java.io.IOException;
public class FileUpload {
/**
* @param file //文件对象
* @param path //上传路径
* @param name //文件名
* @return 文件名
*/
public static String fileUp(MultipartFile file, String path, String name){
//获取文件在服务器的储存位置
File filePath = new File(path);
if (!filePath.exists() && !filePath.isDirectory()) {
filePath.mkdir();
}
//获取原始文件名称(包含格式)
String originalFileName = file.getOriginalFilename();
//获取文件类型,以最后一个`.`为标识
String type = originalFileName.substring(originalFileName.lastIndexOf(".") + 1);
String fileName = name + "." + type;
//在指定路径下创建一个文件
File targetFile = new File(path, fileName);
//将文件保存到服务器指定位置
try {
file.transferTo(targetFile);
} catch (IOException e) {
e.printStackTrace();
}
return fileName;
}
}
(注意表格中数据格式的处理)