如何使用POI生成Excel模板以及設置樣式?具體設置樣式參考我寫的工具類模板
1.導入pom文件
<!--ApachePOI依賴包--> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>4.0.1</version> </dependency> <!--支持excel 2007版本的操作--> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>4.0.1</version> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>ooxml-schemas</artifactId> <version>1.4</version> </dependency>
2.工具類
package cn.mindgd.util; import lombok.Data; import lombok.extern.slf4j.Slf4j; import org.apache.poi.hssf.usermodel.HSSFDateUtil; import org.apache.poi.hssf.usermodel.HSSFSheet; import org.apache.poi.hssf.usermodel.HSSFWorkbook; import org.apache.poi.hssf.util.HSSFColor; import org.apache.poi.ss.usermodel.*; import org.apache.poi.ss.util.CellRangeAddress; import org.apache.poi.xssf.usermodel.XSSFCell; import org.apache.poi.xssf.usermodel.XSSFRow; import org.apache.poi.xssf.usermodel.XSSFSheet; import org.apache.poi.xssf.usermodel.XSSFWorkbook; import org.springframework.util.Base64Utils; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.net.URLEncoder; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.List; /** * @author JiaXinMa * @description Excel工具類 * @date 2021/4/12 */ @Slf4j @Data public class ExcelUtils { private Workbook workbook; private String pattern;// 日期格式 public ExcelUtils(Workbook workbook) { this.workbook = workbook; } /** * @author: JiaXinMa * @description: 下載2007模板 * @date: 2021/6/8 */ public static void downloadExcel2003(HttpServletResponse response, HttpServletRequest request, List<List<Object>> cells, String fireName, String sheetName) { try { fireName = fireName + ".xls"; ExcelUtils.setResponseAttribute(response, request, fireName); HSSFWorkbook workbook = getHSSFWorkbook(cells, sheetName); ServletOutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); //關閉流 workbook.close();//在servlet 完成生命周期之后servlet 容器會自動關閉流 outputStream.close(); } catch (IOException e) { log.info("下載模板失敗!!"); e.printStackTrace(); } } /** * @author: JiaXinMa * @description: 獲取一個文件對象2003版本 HSSFWorkbook * @date: 2021/6/8 * .xls */ public static HSSFWorkbook getHSSFWorkbook(List<List<Object>> rowsData, String sheetName) { //1.創建一個文件對象HSSFWorkbook HSSFWorkbook workbook = new HSSFWorkbook(); //2.創建表單sheet HSSFSheet sheet = workbook.createSheet(sheetName); //3.數據進入Workbook writeRowsDataOnWorkbook(rowsData, sheet); return workbook; } /** * @author: JiaXinMa * @description: 下載2007模板 * @date: 2021/6/8 */ public static void downloadExcel2007(HttpServletResponse response, HttpServletRequest request, List<List<Object>> cells, String fireName, String sheetName) { try { fireName = fireName + ".xlsx"; ExcelUtils.setResponseAttribute(response, request, fireName); XSSFWorkbook workbook = getXSSFWorkbook(cells, sheetName); ServletOutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); //關閉流 workbook.close();//在servlet 完成生命周期之后servlet 容器會自動關閉流 outputStream.close(); } catch (IOException e) { log.info("下載模板失敗!!"); e.printStackTrace(); } } /** * @author: JiaXinMa * @description: 獲取一個文件對象2007版本 XSSFWorkbook * @date: 2021/6/8 * .xlsx */ public static XSSFWorkbook getXSSFWorkbook(List<List<Object>> rowsData, String sheetName) { //1.創建一個文件對象XSSFWorkbook XSSFWorkbook workbook = new XSSFWorkbook(); //2.創建表單sheet XSSFSheet sheet = workbook.createSheet(sheetName); //3.數據進入Workbook writeRowsDataOnWorkbook(rowsData, sheet); return workbook; } /** * @author: JiaXinMa * @description: 寫數據進入Workbook * @date: 2021/6/8 * 如果想設置樣式可以在這里修改 */ private static void writeRowsDataOnWorkbook(List<List<Object>> rowsData, Sheet sheet) { // 遍歷輸出每行 for (int i = 0; i < rowsData.size(); i++) { // 創建一個row Row row = sheet.createRow(i); // 每一行的數據 List<Object> rowData = rowsData.get(i); // 遍歷生成每個單元格 for (int j = 0; j < rowData.size(); j++) { Cell cell = row.createCell(j); // 設置單元格樣式 //cell.setCellStyle() //單元格賦值 Object cellData = rowData.get(j); setDataOnCell(cell, cellData); } } } /** * @author: JiaXinMa * @description: 導出 MaterielStorage excel文件 * @date: 2021/6/8 * sheet為9列 * 第一行為9個單元格合並為一個單元格,背景為灰色,居中 * 第二行為字體加粗,背景為灰色 * 如果想要設置樣式的可以參考該方法 */ public static void downloadMaterielStorageExcelFile(HttpServletResponse response, HttpServletRequest request, List<List<Object>> rowsData, String fileName, String sheetName) { try { // 創建一個Workbook XSSFWorkbook workbook = new XSSFWorkbook(); // 創建一個Sheet XSSFSheet XSSFSheet sheet = workbook.createSheet(sheetName); // 合並單元格 CellRangeAddress region = new CellRangeAddress(0, 0, 0, 8); sheet.addMergedRegion(region); // 設置列寬 setMaterielStorageColumnWidthOnSheet(sheet); CellStyle titleStyle = workbook.createCellStyle();//標題 CellStyle contentFirstStyle = workbook.createCellStyle();//內容第一行 CellStyle contentStyle = workbook.createCellStyle();//內容 //設置單元格背景 titleStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex()); titleStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND); contentFirstStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex()); contentFirstStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND); // 設置單元格邊框 setBorderOnCell(titleStyle); setBorderOnCell(contentFirstStyle); setBorderOnCell(contentStyle); // 設置單元格字體 setFontStylePlus(workbook, titleStyle, "等線", 20, false); setFontStylePlus(workbook, contentFirstStyle, "等線", 14, true); setFontStylePlus(workbook, contentStyle, "等線", 11, false); // 設置單元格居中 titleStyle.setAlignment(HorizontalAlignment.CENTER); contentFirstStyle.setAlignment(HorizontalAlignment.CENTER); contentStyle.setAlignment(HorizontalAlignment.CENTER); // 遍歷輸出每行 for (int i = 0; i < rowsData.size(); i++) { // 創建一個row XSSFRow row = sheet.createRow(i); // 每一行的數據 List<Object> rowData = rowsData.get(i); // 遍歷生成每個單元格 for (int j = 0; j < rowData.size(); j++) { XSSFCell cell = row.createCell(j); // 設置單元格樣式 if (i == 0) { cell.setCellStyle(titleStyle);//標題樣式 } else if (i == 1) { cell.setCellStyle(contentFirstStyle);//內容第一行樣式 } else { cell.setCellStyle(contentStyle);// 內容樣式 } //單元格賦值 Object cellData = rowData.get(j); setDataOnCell(cell, cellData); } } fileName = fileName + ".xls"; ExcelUtils.setResponseAttribute(response, request, fileName); ServletOutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); //關閉流 workbook.close();//在servlet 完成生命周期之后servlet 容器會自動關閉流 outputStream.close(); } catch (IOException e) { log.info("導出excel失敗"); } } /** * @author: JiaXinMa * @description: 設置列的寬度 * @date: 2021/6/8 * 設置MaterielStorage的Excel文件 */ private static void setMaterielStorageColumnWidthOnSheet(Sheet sheet) { sheet.setColumnWidth(0, 2500); // 第1列的寬度為2000 sheet.setColumnWidth(1, 4500); // 第2列的寬度為4500 sheet.setColumnWidth(2, 4500); // 第3列的寬度為4500 sheet.setColumnWidth(3, 4500); // 第4列的寬度為4500 sheet.setColumnWidth(4, 4500); // 第5列的寬度為4500 sheet.setColumnWidth(5, 4500); // 第6列的寬度為4500 sheet.setColumnWidth(6, 4500); // 第7列的寬度為4500 sheet.setColumnWidth(7, 4500); // 第8列的寬度為4500 sheet.setColumnWidth(8, 13000); // 第9列的寬度為13000 } /** * @author: JiaXinMa * @description: 導出 MaterielStorage匯總 excel文件 * @date: 2021/6/17 * sheet為11列 * 第一行為11個單元格合並為一個單元格,背景為灰色,居中 * 第二行為字體加粗,背景為灰色 * 如果想要設置樣式的可以參考該方法 */ public static void downloadMaterielStorageTotalExcelFile(HttpServletResponse response, HttpServletRequest request, List<List<Object>> rowsData, String fileName, String sheetName) { try { // 創建一個Workbook XSSFWorkbook workbook = new XSSFWorkbook(); // 創建一個Sheet XSSFSheet XSSFSheet sheet = workbook.createSheet(sheetName); // 合並單元格 CellRangeAddress region = new CellRangeAddress(0, 0, 0, 10); sheet.addMergedRegion(region); // 設置列寬 setMaterielStorageTotalColumnWidthOnSheet(sheet); CellStyle titleStyle = workbook.createCellStyle();//標題 CellStyle contentFirstStyle = workbook.createCellStyle();//內容第一行 CellStyle contentStyle = workbook.createCellStyle();//內容 //設置單元格背景 titleStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex()); titleStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND); contentFirstStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex()); contentFirstStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND); // 設置單元格邊框 setBorderOnCell(titleStyle); setBorderOnCell(contentFirstStyle); setBorderOnCell(contentStyle); // 設置單元格字體 setFontStylePlus(workbook, titleStyle, "等線", 20, false); setFontStylePlus(workbook, contentFirstStyle, "等線", 14, true); setFontStylePlus(workbook, contentStyle, "等線", 11, false); // 設置單元格居中 titleStyle.setAlignment(HorizontalAlignment.CENTER); contentFirstStyle.setAlignment(HorizontalAlignment.CENTER); contentStyle.setAlignment(HorizontalAlignment.CENTER); // 遍歷輸出每行 for (int i = 0; i < rowsData.size(); i++) { // 創建一個row XSSFRow row = sheet.createRow(i); // 每一行的數據 List<Object> rowData = rowsData.get(i); // 遍歷生成每個單元格 for (int j = 0; j < rowData.size(); j++) { XSSFCell cell = row.createCell(j); // 設置單元格樣式 if (i == 0) { cell.setCellStyle(titleStyle);//標題樣式 } else if (i == 1) { cell.setCellStyle(contentFirstStyle);//內容第一行樣式 } else { cell.setCellStyle(contentStyle);// 內容樣式 } //單元格賦值 Object cellData = rowData.get(j); setDataOnCell(cell, cellData); } } fileName = fileName + ".xls"; ExcelUtils.setResponseAttribute(response, request, fileName); ServletOutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); //關閉流 workbook.close();//在servlet 完成生命周期之后servlet 容器會自動關閉流 outputStream.close(); } catch (IOException e) { log.info("導出excel失敗"); } } /** * @author: JiaXinMa * @description: 設置列的寬度 * @date: 2021/6/17 * 設置MaterielStorage匯總的Excel文件 */ private static void setMaterielStorageTotalColumnWidthOnSheet(Sheet sheet) { sheet.setColumnWidth(0, 2500); // 第1列的寬度為2000 for (int i = 1; i < 11; i++) {// 第2-11列的寬度為4500 sheet.setColumnWidth(i, 4500); } } /** * @author: JiaXinMa * @description: 設置列的寬度 * @date: 2021/6/8 * Integer[] rowColumnWidth 每一列的寬度,有幾列就有幾個長度 */ private static void setColumnWidthOnSheetPlus(Sheet sheet, Integer[] rowColumnWidth) { for (int i = 0; i < rowColumnWidth.length; i++) { sheet.setColumnWidth(i, rowColumnWidth[i]); } } /** * @author: JiaXinMa * @description: 設置字體 * @date: 2021/6/8 */ private static void setFontStylePlus(Workbook workbook, CellStyle cellStyle, String fontName, int fontSize, Boolean isBold) { Font font = workbook.createFont(); font.setColor((short) 42); // 設置字體顏色 font.setColor(HSSFColor.HSSFColorPredefined.BLACK.getColor().getIndex()); // XSSFColor中未找到顏色和short數值的映射,使用HSSFColor來定位顏色的short值 font.setFontName(fontName); // 設置字體 font.setFontHeightInPoints((short) fontSize);// 設置字體大小 font.setBold(isBold);//設置粗體顯示 // font.setUnderline(HSSFFont.SYMBOL_CHARSET);//設置文字下划線 cellStyle.setFont(font); } /** * @author: JiaXinMa * @description: 設置字體 * @date: 2021/6/8 */ private static void setFontStyle(Workbook workbook, CellStyle cellStyle) { Font font = workbook.createFont(); font.setColor((short) 42); // 設置字體顏色 font.setColor(HSSFColor.HSSFColorPredefined.BLACK.getColor().getIndex()); // XSSFColor中未找到顏色和short數值的映射,使用HSSFColor來定位顏色的short值 font.setFontName("宋體"); // 設置字體 font.setFontHeightInPoints((short) 12);// 設置字體大小 // font.setBold(true);//設置粗體顯示 // font.setUnderline(HSSFFont.SYMBOL_CHARSET);//設置文字下划線 cellStyle.setFont(font); } /** * @author: JiaXinMa * @description: 給每一行賦值和樣式 * @date: 2021/6/7 */ private static void setDataAndStyleOnRow(CellStyle titleStyle, Row row, List<Object> rowData) { for (int j = 0; j < rowData.size(); j++) { //給單元格設置樣式 Cell cell = row.createCell(j); cell.setCellStyle(titleStyle); //給單元格賦值 Object cellData = rowData.get(j); setDataOnCell(cell, cellData); } } /** * @author: JiaXinMa * @description: 給單元格設置邊框 * @date: 2021/6/7 */ private static void setBorderOnCell(CellStyle cellStyle) { cellStyle.setBorderBottom(BorderStyle.THIN); // 下邊框 cellStyle.setBorderLeft(BorderStyle.THIN);// 左邊框 cellStyle.setBorderTop(BorderStyle.THIN);// 上邊框 cellStyle.setBorderRight(BorderStyle.THIN);// 右邊框 } /** * @author: JiaXinMa * @description: 給單元格賦值 * @date: 2021/6/7 */ private static void setDataOnCell(Cell cell, Object cellData) { // 如果為空,就不做設值處理 if (cellData == null) { return; } // String類型數值 if (cellData.getClass() == String.class) { cell.setCellValue((String) cellData); } // double類型數值 else if (cellData.getClass() == double.class || cellData.getClass() == Double.class) { cell.setCellValue((Double) cellData); } // float類型數值 else if (cellData.getClass() == float.class || cellData.getClass() == Float.class) { cell.setCellValue((Float) cellData); } // integer類型數值 else if (cellData.getClass() == int.class || cellData.getClass() == Integer.class) { cell.setCellValue((Integer) cellData); } } /** * @author: JiaXinMa * @description: 設置響應頭的屬性 * @date: 2021/6/8 */ public static void setResponseAttribute(HttpServletResponse response, HttpServletRequest request, String fileName) throws UnsupportedEncodingException { response.setContentType("application/json"); response.setCharacterEncoding("utf-8"); String agent = request.getHeader("USER-AGENT").toLowerCase(); String uncodeFileNameString = ""; if (agent.contains("firefox")) {//判斷是否火狐 uncodeFileNameString = "=?UTF-8?B?" + (new String(Base64Utils.encodeToString(fileName.getBytes("UTF-8")))) + "?="; } else { uncodeFileNameString = URLEncoder.encode(fileName, "UTF-8"); uncodeFileNameString = uncodeFileNameString.replace("+", "%20"); // 空格被轉為了 "+" ,要轉成utf-8的空格符號 "%20" } response.setContentType("application/octet-stream;charset=UTF-8"); response.setHeader("Content-disposition", "attachment; filename=" + uncodeFileNameString); } /** * 是否是2003的excel,返回true是2003 */ public static boolean isExcel2003(String filePath) { return filePath.matches("^.+\\.(?i)(xls)$"); } /** * 是否是2007的excel,返回true是2007 */ public static boolean isExcel2007(String filePath) { return filePath.matches("^.+\\.(?i)(xlsx)$"); } /** * 驗證EXCEL文件 */ public static boolean validateExcel(String filePath) { if (filePath == null || !(isExcel2003(filePath) || isExcel2007(filePath))) { return false; } return true; } /** * 讀取 Excel 第一頁數據 */ public List<List<String>> read() throws Exception { return read(0, 0, getRowCount(0) - 1); } /** * 讀取 Excel 數據 * 指定 sheet 頁,從 0 開始 */ public List<List<String>> read(int sheetIx) throws Exception { return read(sheetIx, 0, getRowCount(sheetIx) - 1); } /** * 讀取 Excel 數據 * 指定 sheet 頁,從 0 開始 * 指定開始行,從 0 開始 * 指定結束行,從 0 開始 */ public List<List<String>> read(int sheetIx, int start, int end) throws Exception { Sheet sheet = workbook.getSheetAt(sheetIx); List<List<String>> list = new ArrayList<List<String>>(); if (end > getRowCount(sheetIx)) { end = getRowCount(sheetIx); } int cols = sheet.getRow(0).getLastCellNum(); // 第一行總列數 for (int i = start; i <= end; i++) { List<String> rowList = new ArrayList<String>(); Row row = sheet.getRow(i); for (int j = 0; j < cols; j++) { if (row == null) { rowList.add(null); continue; } rowList.add(getCellValueToString(row.getCell(j))); } list.add(rowList); } return list; } /** * 返回sheet中的行數 */ public int getRowCount(int sheetIx) { Sheet sheet = workbook.getSheetAt(sheetIx); if (sheet.getPhysicalNumberOfRows() == 0) { return 0; } return sheet.getLastRowNum() + 1; } /** * 將轉換單元格的類型為String */ private String getCellValueToString(Cell cell) { String strCell = ""; if (cell == null) { return null; } switch (cell.getCellType()) { case BOOLEAN: strCell = String.valueOf(cell.getBooleanCellValue()); break; case NUMERIC: if (HSSFDateUtil.isCellDateFormatted(cell)) { Date date = cell.getDateCellValue(); if (pattern != null) { SimpleDateFormat sdf = new SimpleDateFormat(pattern); strCell = sdf.format(date); } else { strCell = date.toString(); } break; } // 不是日期格式,則防止當數字過長時以科學計數法顯示 cell.setCellType(CellType.STRING); strCell = cell.toString(); break; case STRING: strCell = cell.getStringCellValue(); break; default: break; } return strCell; } }
3.具體業務層和控制層我就不列代碼了,直接調用工具類的方法就行
效果如下:
控制層最好就是無返回值
如果想讀取excel文件可以調用工具類的 read()方法,大家可以結合自己的業務多嘗試寫一些工具類,一起學習交流
想看更多精彩內容,可以關注我的CSDN