前言
上一章作者分享了mybatis通用mapper的實(shí)用覆旭。在實(shí)際開發(fā)中埠褪,我們常常會用到excel導(dǎo)入導(dǎo)出相關(guān)的功能,本章宫仗,作者將分享一個實(shí)用的excel操作案例够挂,實(shí)現(xiàn)excel的導(dǎo)入導(dǎo)出功能。
亮點(diǎn):
- excel導(dǎo)入反射java實(shí)體
- excel導(dǎo)出使用模版
這邊簡單解釋下藕夫。
excel導(dǎo)入反射java實(shí)體:
首先我們講下正常實(shí)現(xiàn)excel導(dǎo)入的方法孽糖,如下偽代碼
public List<Student> importExcel(File excelFile, Student student){
List<Student> list = new ArrayList<Student>();
for (excelFile) {
Student s = new Student();
//excel的第一行的第一列為name
student.name = excelFile.row[0].column[0];
//excel的第一行的第二列為age
student.age = excelFile.row[0].column[1];
list.add(s);
}
return list;
}
可以看到,我們實(shí)現(xiàn)了將excel數(shù)據(jù)列轉(zhuǎn)換為student實(shí)體的邏輯毅贮。那么办悟,問題來了,以后我想轉(zhuǎn)為teacher實(shí)體怎么辦滩褥?在寫一個類似的方法病蛉?可以看出,這種方式在方法體內(nèi)部實(shí)現(xiàn)實(shí)體類屬性與excel列的綁定,很不靈活铺然。那么俗孝,能不能傳入一個實(shí)體類,在運(yùn)行時魄健,動態(tài)的獲取該實(shí)體的屬性實(shí)現(xiàn)綁定呢赋铝?。此時就很有必要了解java的反射機(jī)制了沽瘦。
JAVA反射機(jī)制是在運(yùn)行狀態(tài)中革骨,對于任意一個實(shí)體類,都能夠知道這個類的所有屬性和方法其垄;對于任意一個對象苛蒲,都能夠調(diào)用它的任意方法和屬性卤橄。
我們再看看利用反射機(jī)制實(shí)現(xiàn)excel導(dǎo)入
public <T extends Object> List<T> importExcel(File excelFile, Class<T> clazz) {
List<T> list = new ArrayList<>();
//獲取類的所有屬性
Field[] fields = clazz.getDeclaredFields();
T obj = null;
for (excelFile){
// 創(chuàng)建實(shí)體
obj = clazz.newInstance();
for (Field f : fields) {
//屬性賦值绿满,實(shí)現(xiàn)屬性和excel列的綁定關(guān)系
}
list.add(obj);
}
return list;
}
可以看到,利用反射機(jī)制實(shí)現(xiàn)的導(dǎo)入方法更加靈活窟扑,這樣就可以使其更加通用喇颁。
excel導(dǎo)出使用模版
實(shí)現(xiàn)過excel導(dǎo)出功能的可能都知道,excel最麻煩的就是編寫樣式嚎货,尤其是樣式復(fù)雜的表格橘霎。仔細(xì)觀察表格,你會發(fā)現(xiàn)殖属,正常復(fù)雜的都是表頭樣式姐叁,往往是這樣的(舉個簡單的例子)
然后下半部分可能就是正常的列表展示。
那么我們能不能把這復(fù)雜的部分抽出來洗显,單獨(dú)由人工來做(畢竟office建立這么一個表格so easy外潜,用代碼去畫簡直抓狂。)挠唆,而后处窥,我們代碼讀取這個表格,動態(tài)填充數(shù)據(jù)呢玄组?答案滔驾,是肯定的。在接下來的案例中你會看到這樣的實(shí)現(xiàn)俄讹。
目錄結(jié)構(gòu)
添加依賴
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.0.0</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.15</version>
</dependency>
<!-- SpringBoot通用Mapper依賴 -->
<dependency>
<groupId>tk.mybatis</groupId>
<artifactId>mapper-spring-boot-starter</artifactId>
<version>2.1.5</version>
</dependency>
<!-- excel依賴 -->
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>3.17</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>3.17</version>
</dependency>
<!-- 實(shí)用工具 -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<scope>provided</scope>
</dependency>
</dependencies>
這里我們除了使用了poi excel的依賴哆致,還使用了上章用過的通用mapper依賴,以及l(fā)ombok依賴患膛,關(guān)于lombok依賴摊阀,你可以簡單理解為一個工具包,這樣我們編寫實(shí)體類時只需要使用
@Setter
@Getter
兩個注解即可自動實(shí)現(xiàn)類屬性的set和get方法,省了不少事驹溃。
添加配置
spring:
datasource:
url: jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&characterEncoding=UTF-8&useSSL=false&autoReconnect=true&failOverReadOnly=false&serverTimezone=GMT%2B8
username: root
password: 123456
driver-class-name: com.mysql.cj.jdbc.Driver
完善代碼
我們先實(shí)現(xiàn)excel的工具類
utils/excel/ExcelAnnotation
package com.mrcoder.sbmexcel.utils.excel;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
@Retention(RetentionPolicy.RUNTIME)
public @interface ExcelAnnotation {
/**
* 列索引
*
* @return
*/
public int columnIndex() default 0;
/**
* 列名
*
* @return
*/
public String columnName() default "";
}
ExcelAnnotation這個類實(shí)現(xiàn)了一個自定義的注解城丧,定義了兩個屬性,這樣我們可以在實(shí)體類中使用
//columnIndex定義excel的列豌鹤,columnName定義表列名稱
@ExcelAnnotation(columnIndex = 1, columnName = "姓名")
private String name;
意思就是導(dǎo)入的excel第二列對應(yīng)數(shù)據(jù)表的name列亡哄。
此處注意,excel索引是從0開始的布疙。
我們看下需要導(dǎo)入的excel表格結(jié)構(gòu)
第一列需要是我們無需導(dǎo)入的蚊惯,所以,我們定義實(shí)體類時灵临,excel注解從1開始截型。
接下來我們定義實(shí)體類
model/Excel
package com.mrcoder.sbmexcel.model;
import com.mrcoder.sbmexcel.utils.excel.ExcelAnnotation;
import lombok.Getter;
import lombok.Setter;
import lombok.experimental.Accessors;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.Table;
import java.util.Date;
@Setter
@Getter
@Table(name = "excel")
@Accessors(chain = true)
public class Excel {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@ExcelAnnotation(columnIndex = 1, columnName = "姓名")
private String name;
@ExcelAnnotation(columnIndex = 2, columnName = "年齡")
private Integer age;
@ExcelAnnotation(columnIndex = 3, columnName = "身高")
private Double height;
@ExcelAnnotation(columnIndex = 4, columnName = "體重")
private Double weight;
@ExcelAnnotation(columnIndex = 5, columnName = "學(xué)歷")
private String edu;
private Date createTime;
private Date updateTime;
private Integer status;
}
從實(shí)體類中可以看出,我們使用了
@Setter
@Getter
lombok注解儒溉,所以在該實(shí)體中無需手寫get和set方法宦焦。
然后我們實(shí)現(xiàn)excel讀取反射成實(shí)體的工具類
utils/excel/ExcelUtil
package com.mrcoder.sbmexcel.utils.excel;
import java.io.*;
import java.lang.reflect.Field;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.List;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.web.multipart.MultipartFile;
/**
* Excel 工具類
*
* @author mrcoder
* @version 1.0 2019.05.29
*/
public class ExcelUtil {
/**
* 讀取excel反射實(shí)體
*
* @param file MultipartFile
* @param clazz entity
* @return
* @throws RuntimeException
*/
public static <T extends Object> List<T> readExcelObject(MultipartFile file, Class<T> clazz) {
// 存儲excel數(shù)據(jù)
List<T> list = new ArrayList<>();
Workbook wookbook = null;
InputStream inputStream = null;
try {
inputStream = file.getInputStream();
} catch (IOException e) {
throw new RuntimeException("文件讀取異常");
}
// 根據(jù)excel文件版本獲取工作簿
if (file.getOriginalFilename().endsWith(".xls")) {
wookbook = xls(inputStream);
} else if (file.getOriginalFilename().endsWith(".xlsx")) {
wookbook = xlsx(inputStream);
} else {
throw new RuntimeException("非excel文件");
}
// 得到一個工作表
Sheet sheet = wookbook.getSheetAt(0);
// 獲取行總數(shù)
int rows = sheet.getLastRowNum() + 1;
Row row;
// 獲取類所有屬性
Field[] fields = clazz.getDeclaredFields();
T obj = null;
int coumnIndex = 0;
Cell cell = null;
ExcelAnnotation excelAnnotation = null;
for (int i = 1; i < rows; i++) {
// 獲取excel行
row = sheet.getRow(i);
//此處用來過濾空行
Cell cell0 = row.getCell(0);
cell0.setCellType(CellType.STRING);
Cell cell1 = row.getCell(1);
cell1.setCellType(CellType.STRING);
if (cell0.getStringCellValue() == "" && cell1.getStringCellValue() == "") {
continue;
}
try {
// 創(chuàng)建實(shí)體
obj = clazz.newInstance();
for (Field f : fields) {
// 設(shè)置屬性可訪問
f.setAccessible(true);
// 判斷是否是注解
if (f.isAnnotationPresent(ExcelAnnotation.class)) {
// 獲取注解
excelAnnotation = f.getAnnotation(ExcelAnnotation.class);
// 獲取列索引
coumnIndex = excelAnnotation.columnIndex();
// 獲取單元格
cell = row.getCell(coumnIndex);
// 設(shè)置屬性
setFieldValue(obj, f, wookbook, cell);
}
}
System.out.println(obj);
// 添加到集合中
list.add(obj);
} catch (InstantiationException e1) {
e1.printStackTrace();
} catch (IllegalAccessException e1) {
e1.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("excel文件內(nèi)容出錯");
}
}
try {
//釋放資源
wookbook.close();
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
return list;
}
/**
* 綁定實(shí)體值
*
* @param obj Object
* @param f Field
* @param wookbook Workbook
* @param cell Cell
* @return
* @throws RuntimeException
*/
private static void setFieldValue(Object obj, Field f, Workbook wookbook, Cell cell) {
try {
cell.setCellType(CellType.STRING);
if (f.getType() == byte.class || f.getType() == Byte.class) {
f.set(obj, Byte.parseByte(cell.getStringCellValue()));
} else if (f.getType() == int.class || f.getType() == Integer.class) {
f.set(obj, Integer.parseInt(cell.getStringCellValue()));
} else if (f.getType() == Double.class || f.getType() == double.class) {
f.set(obj, Double.parseDouble(cell.getStringCellValue()));
} else if (f.getType() == BigDecimal.class) {
f.set(obj, new BigDecimal(cell.getStringCellValue()));
} else {
f.set(obj, cell.getStringCellValue());
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 對excel 2003處理
*/
private static Workbook xls(InputStream is) {
try {
// 得到工作簿
return new HSSFWorkbook(is);
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
/**
* 對excel 2007處理
*/
private static Workbook xlsx(InputStream is) {
try {
// 得到工作簿
return new XSSFWorkbook(is);
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
}
該工具類接受上傳excel文件,以及實(shí)體類參數(shù)顿涣,主要就是將每一行的excel數(shù)據(jù)轉(zhuǎn)換成一個實(shí)體波闹,這樣方便我們后續(xù)批量入庫操作。
mapper/ExcelMapper
package com.mrcoder.sbmexcel.mapper;
import com.mrcoder.sbmexcel.model.Excel;
import tk.mybatis.mapper.additional.insert.InsertListMapper;
import tk.mybatis.mapper.common.ExampleMapper;
import tk.mybatis.mapper.common.Mapper;
public interface ExcelMapper extends Mapper<Excel>, InsertListMapper<Excel>, ExampleMapper<Excel> {
}
service/ExcelService
package com.mrcoder.sbmexcel.service;
import com.mrcoder.sbmexcel.mapper.ExcelMapper;
import com.mrcoder.sbmexcel.model.Excel;
import com.mrcoder.sbmexcel.utils.excel.ExcelUtil;
import org.apache.poi.ss.usermodel.BorderStyle;
import org.apache.poi.ss.usermodel.CellStyle;
import org.apache.poi.ss.usermodel.FillPatternType;
import org.apache.poi.ss.usermodel.IndexedColors;
import org.apache.poi.ss.util.CellRangeAddress;
import org.apache.poi.xssf.usermodel.XSSFCellStyle;
import org.apache.poi.xssf.usermodel.XSSFRow;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.util.ResourceUtils;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.MultipartHttpServletRequest;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
@Service
public class ExcelService {
@Autowired
private ExcelMapper excelMapper;
//excel導(dǎo)入
public int importExcel(HttpServletRequest request) {
MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;
MultipartFile excel = multipartRequest.getFile("filename");
try {
List<Excel> excelData = ExcelUtil.readExcelObject(excel, Excel.class);
//檢查每列數(shù)據(jù)
for (int i = 0; i < excelData.size(); i++) {
excelData.get(i).setStatus(1);
Date time = new Date();
excelData.get(i).setCreateTime(time);
excelData.get(i).setUpdateTime(time);
}
//批量導(dǎo)入
return excelMapper.insertList(excelData);
} catch (Exception e) {
throw new RuntimeException(e.getMessage());
}
}
//excel導(dǎo)入
public void exportExcel(HttpServletResponse response) {
try {
// 1.讀取Excel模板
File file = ResourceUtils.getFile(ResourceUtils.CLASSPATH_URL_PREFIX + "excel/export.xlsx");
InputStream in = new FileInputStream(file);
XSSFWorkbook wb = new XSSFWorkbook(in);
// 2.讀取模板里面的所有Sheet
XSSFSheet sheet = wb.getSheetAt(0);
// 3.設(shè)置公式自動讀取
sheet.setForceFormulaRecalculation(true);
// 4.向相應(yīng)的單元格里面設(shè)置值
XSSFRow row;
// 5.得到第二行
row = sheet.getRow(1);
// 6.設(shè)置單元格屬性值和樣式
row.getCell(1).setCellValue("張三");
row.getCell(3).setCellValue("18");
row.getCell(6).setCellValue("本科");
row.getCell(8).setCellValue(new Date());
row = sheet.getRow(2);
row.getCell(1).setCellValue("1511xxxx234");
row.getCell(3).setCellValue("廣東");
row.getCell(6).setCellValue("本科");
row = sheet.getRow(3);
row.getCell(1).setCellValue("180");
row.getCell(3).setCellValue("已婚");
row.getCell(6).setCellValue("深圳");
row.getCell(8).setCellValue("2");
row = sheet.getRow(4);
row.getCell(1).setCellValue("60");
row.getCell(3).setCellValue("中國");
row.getCell(6).setCellValue("其它");
row.getCell(8).setCellValue("備注");
//單元格合并
row = sheet.getRow(6);
row.getCell(0).setCellValue("合并列");
CellRangeAddress region = new CellRangeAddress(6, 6, 0, 5);
sheet.addMergedRegion(region);
//單元格設(shè)置背景色
CellStyle style = wb.createCellStyle();
style.setFillForegroundColor(IndexedColors.SKY_BLUE.getIndex());
style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
row.getCell(0).setCellStyle(style);
//設(shè)置單元格邊框
row = sheet.getRow(8);
XSSFCellStyle style2 = wb.createCellStyle();
style2.setBorderBottom(BorderStyle.DOUBLE);
style2.setBorderRight(BorderStyle.DOUBLE);
style2.setBorderLeft(BorderStyle.DOUBLE);
style2.setBorderTop(BorderStyle.DOUBLE);
style2.setBottomBorderColor(IndexedColors.SKY_BLUE.getIndex());
style2.setRightBorderColor(IndexedColors.SKY_BLUE.getIndex());
row.getCell(0).setCellStyle(style2);
// 7.設(shè)置sheet名稱和單元格內(nèi)容
wb.setSheetName(0, "測試");
String fileName = new String(("export-" + new SimpleDateFormat("yyyy-MM-dd").format(new Date()))
.getBytes(), "UTF-8");
response.setContentType("multipart/form-data");
response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xlsx");
response.setHeader("Pragma", "No-cache");
response.setCharacterEncoding("utf-8");
ServletOutputStream out = response.getOutputStream();
wb.write(out);
out.flush();
out.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
ExcelService實(shí)現(xiàn)了導(dǎo)入導(dǎo)出功能的封裝涛碑。導(dǎo)入功能無需多說精堕,導(dǎo)出實(shí)現(xiàn)了讀取模版,填充數(shù)據(jù)蒲障,以及對Excel 表格的一些合并歹篓,樣式修改等功能~
最后,我們完成controller
ExcelController
package com.mrcoder.sbmexcel.controller;
import com.mrcoder.sbmexcel.service.ExcelService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@RestController
public class ExcelController {
@Autowired
private ExcelService excelService;
@PostMapping("/importExcel")
public int importExcel(HttpServletRequest request) {
return excelService.importExcel(request);
}
@GetMapping("/exportExcel")
public void exportExcel(HttpServletResponse response) {
excelService.exportExcel(response);
}
}
運(yùn)行
http://localhost:8080/importExcel 導(dǎo)入
http://localhost:8080/exportExcel 導(dǎo)出
項目地址
https://github.com/MrCoderStack/SpringBootDemo/tree/master/sbm-excel