导出Excel的技术分享-综合篇

在这里插入图片描述

导出Excel的技术分享-综合篇

简单的EasyExcel使用

/*** 最简单的写*/public void simpleWrite() {// 注意 simpleWrite在数据量不大的情况下可以使用(5000以内,具体也要看实际情况),数据量大参照 重复多次写入// 写法1 JDK8+// since: 3.0.0-beta1String fileName = "simpleWrite" + System.currentTimeMillis() + ".xlsx";// 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭// 如果这里想使用03 则 传入excelType参数即可EasyExcel.write(fileName, DemoData.class).sheet("模板").doWrite(data());}private List<DemoData> data() {List<DemoData> list = ListUtils.newArrayList();for (int i = 0; i < 10; i++) {DemoData data = new DemoData();data.setString("字符串" + i);data.setDate(new Date());data.setDoubleData(0.56);list.add(data);}return list;}

行标题设置

   public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("行标题设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"行标题设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class IndexStandardStyleHandler implements SheetWriteHandler {private String tableTitle;public IndexStandardStyleHandler(String title) {this.tableTitle = title;}@Overridepublic void afterSheetCreate(SheetWriteHandlerContext context) {// 在创建工作表之后执行的操作WriteSheetHolder writeSheetHolder = context.getWriteSheetHolder();
//        ExcelWriteHeadProperty excelWriteHeadProperty = writeSheetHolder.getExcelWriteHeadProperty();
//        excelWriteHeadProperty.setHeadRowNumber(3);// 设置行高和样式// 创建第一行并设置样式Row row = writeSheetHolder.getSheet().createRow(0);row.setHeight((short) 1000);Cell cell = row.createCell(0);cell.setCellValue(tableTitle);//设置标题// 准备样式CellStyle cellStyle = context.getWriteWorkbookHolder().getWorkbook().createCellStyle();cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);cellStyle.setAlignment(HorizontalAlignment.CENTER);Font font = context.getWriteWorkbookHolder().getWorkbook().createFont();
//        font.setBold(true);font.setFontHeight((short) 500);font.setFontName("宋体");font.setColor(HSSFColor.HSSFColorPredefined.RED.getIndex());cellStyle.setFont(font);cellStyle.setAlignment(HorizontalAlignment.CENTER);cellStyle.setWrapText(true);cell.setCellStyle(cellStyle);//writeWorkbookHolder.createCellStyle(style,null);Sheet sheet = context.getWriteSheetHolder().getSheet();sheet.addMergedRegionUnsafe(new CellRangeAddress(0, 1, 0, 2));}@Overridepublic int order() {return 3;}
}

单元格线设置

 public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("单元格线设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"单元格线设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class CelllineStyleHandler implements CellWriteHandler {@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {for (int i = 0; i < cellDataList.size(); i++) {WriteCellData<?> writeCellData = cellDataList.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setShrinkToFit(true);}}@Overridepublic int order() {return 1;}
}

下拉框设置

  public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("下拉框设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"下拉框设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new CellDropDownBoxWriteHandler()).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class CellDropDownBoxWriteHandler implements SheetWriteHandler {/*** 定义一个map key是需要添加下拉框的列的index value是下拉框数据*/Map<Integer, String[]> mapDropDown = new HashMap<>();String[] checkFrequency = {"字符串0","字符串1","字符串2","字符串3","字符串4"};@Overridepublic void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {//下拉选在Excel中对应的列Field[] fields = writeSheetHolder.getClazz().getDeclaredFields();for (int i = 0; i < fields.length; i++) {Field field = fields[i];if (field.isAnnotationPresent(ExcelProperty.class) && field.isAnnotationPresent(ExcelProperty.class)){mapDropDown.put(i,checkFrequency);}}//获取工作簿Sheet sheet = writeSheetHolder.getSheet();///开始设置下拉框DataValidationHelper helper = sheet.getDataValidationHelper();//设置下拉框for (Map.Entry<Integer, String[]> entry : mapDropDown.entrySet()) {/*起始行、终止行、起始列、终止列  起始行为1即表示表头不设置**///这里设置65535可能又问题,因为这个是excel的最大行数,如果数据量超过这个数,就会报错CellRangeAddressList addressList = new CellRangeAddressList(1, 65535, entry.getKey(), entry.getKey());/*设置下拉框数据**/DataValidationConstraint constraint = helper.createExplicitListConstraint(entry.getValue());DataValidation dataValidation = helper.createValidation(constraint, addressList);//阻止输入非下拉选项的值dataValidation.setErrorStyle(DataValidation.ErrorStyle.STOP);dataValidation.setShowErrorBox(true);dataValidation.setSuppressDropDownArrow(true);dataValidation.createErrorBox("提示", "输入值与单元格定义格式不一致");dataValidation.createPromptBox("填写说明", "填写内容只能为下拉数据集中的类型");sheet.addValidationData(dataValidation);}}@Overridepublic int order() {return 1;}
}

标题和内容字体样式设置不一样

 public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("标题和内容字体样式不一样设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"标题和内容字体样式不一样设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new CellDropDownBoxWriteHandler()).registerWriteHandler(new FontStyleWriteHandler()).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class FontStyleWriteHandler implements CellWriteHandler {/*** 设置字体样式* @param writeSheetHolder* @param writeTableHolder Nullable.It is null without using table writes.* @param cellDataList     Nullable.It is null in the case of add header.There may be several when fill the data.* @param cell* @param head             Nullable.It is null in the case of fill data and without head.* @param relativeRowIndex Nullable.It is null in the case of fill data.* @param isHead           It will always be false when fill data.*/@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {for (int i = 0; i < cellDataList.size(); i++) {WriteCellData<?> writeCellData = cellDataList.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();if (2 == cell.getRowIndex()) {WriteFont writeFont = new WriteFont();writeFont.setBold(true);writeFont.setFontHeightInPoints((short) 13);writeFont.setFontName("楷体");writeCellStyle.setWriteFont(writeFont);} else {WriteFont writeFont = new WriteFont();writeFont.setFontName("宋体");writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setWriteFont(writeFont);
//                writeCellStyle.setShrinkToFit(true);}}}@Overridepublic int order() {return 1;}
}

单元格合并设置

 public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("单元格合并设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"单元格合并设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new CellDropDownBoxWriteHandler()).registerWriteHandler(new FontStyleWriteHandler()).registerWriteHandler(new ExcelFillCellMergeStrategyHandler(2,new int[]{0,1,2})).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class ExcelFillCellMergeStrategyHandler implements CellWriteHandler {/*** 合并字段的下标,如第一到五列new int[]{0,1,2,3,4}*/private int[] mergeColumnIndex;/*** 从第几行开始合并,如果表头占两行,这个数字就是2*/private int mergeRowIndex;public ExcelFillCellMergeStrategyHandler(int mergeRowIndex, int[] mergeColumnIndex) {this.mergeRowIndex = mergeRowIndex;this.mergeColumnIndex = mergeColumnIndex;}@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder,List<WriteCellData<?>> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {//当前行int curRowIndex = cell.getRowIndex();//当前列int curColIndex = cell.getColumnIndex();if (curRowIndex > mergeRowIndex) {for (int i = 0; i < mergeColumnIndex.length; i++) {if (curColIndex == mergeColumnIndex[i]) {mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);break;}}}for (int i = 0; i < list.size(); i++) {WriteCellData<?> writeCellData = list.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setShrinkToFit(true);}}private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {//获取当前行的当前列的数据和上一行的当前列列数据,通过上一行数据是否相同进行合并Object curData = cell.getCellType() == CellType.STRING ? cell.getStringCellValue() :cell.getNumericCellValue();Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);Object preData = preCell.getCellType() == CellType.STRING ? preCell.getStringCellValue() :preCell.getNumericCellValue();// 比较当前行的第一列的单元格与上一行是否相同,相同合并当前单元格与上一行//if (curData.equals(preData)) {Sheet sheet = writeSheetHolder.getSheet();List<CellRangeAddress> mergeRegions = sheet.getMergedRegions();boolean isMerged = false;for (int i = 0; i < mergeRegions.size() && !isMerged; i++) {CellRangeAddress cellRangeAddr = mergeRegions.get(i);// 若上一个单元格已经被合并,则先移出原有的合并单元,再重新添加合并单元if (cellRangeAddr.isInRange(curRowIndex - 1, curColIndex)) {sheet.removeMergedRegion(i);cellRangeAddr.setLastRow(curRowIndex);sheet.addMergedRegion(cellRangeAddr);isMerged = true;}}// 若上一个单元格未被合并,则新增合并单元if (!isMerged) {CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, curColIndex,curColIndex);sheet.addMergedRegion(cellRangeAddress);}}}@Overridepublic int order() {return 1;}
}

自定义多个sheet页导出设置

public class Test07 {public static void main(String[] args) throws NoSuchFieldException, ClassNotFoundException, IllegalAccessException {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("自定义多个sheet页导出设置.xlsx").build();// 第一个sheet的写入WriteSheet writeSheet01 = EasyExcel.writerSheet(0,"自定义多个sheet页导出设置").head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new CellDropDownBoxWriteHandler()).registerWriteHandler(new FontStyleWriteHandler()).registerWriteHandler(new ExcelFillCellMergeStrategyHandler(2,new int[]{0,1,2})).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet01);// 后续sheet写入Map map = listGroupInfo(DemoData.class, dataList());int i = 1;Map<String,String> orgNameMap = new HashMap<>();orgNameMap.put("自定义sheet页1","自定义sheet页1");orgNameMap.put("自定义sheet页2","自定义sheet页2");orgNameMap.put("自定义sheet页3","自定义sheet页3");orgNameMap.put("自定义sheet页4","自定义sheet页4");for (Map.Entry<String, String> entry : orgNameMap.entrySet()) {String OrgCode = entry.getKey();String orgName = entry.getValue();Object list = map.get(OrgCode);// 设置非供电单位写入标题String noPowerTitle = "行单元格样例演示\n" +OrgCode+"行单元格样例演示\n";;WriteSheet sheet = EasyExcel.writerSheet(i, orgName).head(DemoData.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).build();excelWriter.write((Collection<?>) list,sheet);i = i + 1;}excelWriter.finish();}private static Map listGroupInfo(Class<?> head1, List<?> dataList1) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException {String groupField = "string";String name = head1.getName();Class<?> clazz = Class.forName(name);Field field = clazz.getDeclaredField(groupField);field.setAccessible(true);Map<Object, ? extends List<?>> map = dataList1.stream().collect(Collectors.groupingBy(obj -> {try {return field.get(obj);} catch (IllegalAccessException e) {e.printStackTrace();return null;}}));Set<Object> objects = map.keySet();Iterator<Object> iterator = objects.iterator();
//        while (iterator.hasNext()) {
//            List<?> list = map.get(iterator.next());
//            for (int i = 0; i < list.size(); i++) {
//                Object o = list.get(i);
//                Field fieldSerialNumber = clazz.getDeclaredField("serialNumber");
//                fieldSerialNumber.setAccessible(true);
//                fieldSerialNumber.set(o,String.valueOf(i+1));
//            }
//        }return map;}private static List<DemoData> data() {List<DemoData> list = ListUtils.newArrayList();for (int i = 0; i < 10; i++) {DemoData data = new DemoData();data.setString("字符串" + i);data.setDate(new Date());data.setDoubleData(0.56);list.add(data);}return list;}private static List<DemoData> dataList() {List<DemoData> list = ListUtils.newArrayList();for (int i = 0; i < 10; i++) {DemoData data1 = new DemoData();data1.setString("自定义sheet页1");data1.setDate(new Date());data1.setDoubleData(111.111);list.add(data1);}for (int i = 0; i < 10; i++) {DemoData data2 = new DemoData();data2.setString("自定义sheet页2");data2.setDate(new Date());data2.setDoubleData(222.222);list.add(data2);}for (int i = 0; i < 10; i++) {DemoData data3 = new DemoData();data3.setString("自定义sheet页3");data3.setDate(new Date());data3.setDoubleData(333.333);list.add(data3);}for (int i = 0; i < 10; i++) {DemoData data3 = new DemoData();data3.setString("自定义sheet页4");data3.setDate(new Date());data3.setDoubleData(444.444);list.add(data3);}return list;}}
public class ExcelFillCellMergeStrategyHandler implements CellWriteHandler {/*** 合并字段的下标,如第一到五列new int[]{0,1,2,3,4}*/private int[] mergeColumnIndex;/*** 从第几行开始合并,如果表头占两行,这个数字就是2*/private int mergeRowIndex;public ExcelFillCellMergeStrategyHandler(int mergeRowIndex, int[] mergeColumnIndex) {this.mergeRowIndex = mergeRowIndex;this.mergeColumnIndex = mergeColumnIndex;}@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder,List<WriteCellData<?>> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {//当前行int curRowIndex = cell.getRowIndex();//当前列int curColIndex = cell.getColumnIndex();if (curRowIndex > mergeRowIndex) {for (int i = 0; i < mergeColumnIndex.length; i++) {if (curColIndex == mergeColumnIndex[i]) {mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);break;}}}for (int i = 0; i < list.size(); i++) {WriteCellData<?> writeCellData = list.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setShrinkToFit(true);}}private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {//获取当前行的当前列的数据和上一行的当前列列数据,通过上一行数据是否相同进行合并Object curData = cell.getCellType() == CellType.STRING ? cell.getStringCellValue() :cell.getNumericCellValue();Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);Object preData = preCell.getCellType() == CellType.STRING ? preCell.getStringCellValue() :preCell.getNumericCellValue();// 比较当前行的第一列的单元格与上一行是否相同,相同合并当前单元格与上一行//if (curData.equals(preData)) {Sheet sheet = writeSheetHolder.getSheet();List<CellRangeAddress> mergeRegions = sheet.getMergedRegions();boolean isMerged = false;for (int i = 0; i < mergeRegions.size() && !isMerged; i++) {CellRangeAddress cellRangeAddr = mergeRegions.get(i);// 若上一个单元格已经被合并,则先移出原有的合并单元,再重新添加合并单元if (cellRangeAddr.isInRange(curRowIndex - 1, curColIndex)) {sheet.removeMergedRegion(i);cellRangeAddr.setLastRow(curRowIndex);sheet.addMergedRegion(cellRangeAddr);isMerged = true;}}// 若上一个单元格未被合并,则新增合并单元if (!isMerged) {CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, curColIndex,curColIndex);sheet.addMergedRegion(cellRangeAddress);}}}@Overridepublic int order() {return 1;}
}

单元格锁设置

 public static void main(String[] args) {String powerTitle = "行单元格样例演示\n" +"行单元格样例演示\n";ExcelWriter excelWriter = EasyExcel.write("单元格锁设置.xlsx").build();WriteSheet writeSheet = EasyExcel.writerSheet(0,"单元格锁设置").head(DemoDataLock.class).relativeHeadRowIndex(2).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new CelllineStyleHandler()).registerWriteHandler(new LockHandler()).registerWriteHandler(new IndexStandardStyleHandler(powerTitle)).build();excelWriter.write(data(),writeSheet);excelWriter.finish();}
public class LockHandler extends LongestMatchColumnWidthStyleStrategy implements SheetWriteHandler {@Overridepublic void afterCellDispose(CellWriteHandlerContext context) {// 锁定有内容过的单元格(方法1)Class clazz = context.getWriteSheetHolder().getClazz();Field[] fields = clazz.getDeclaredFields();Field field = fields[context.getColumnIndex()];if (field.isAnnotationPresent(UnLockCell.class) && field.isAnnotationPresent(ExcelProperty.class)) {WriteCellStyle writeCellStyle = context.getFirstCellData().getOrCreateStyle();writeCellStyle.setLocked(false);}}@Overridepublic void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {// 设置保护密码writeSheetHolder.getSheet().protectSheet("ZhiChengKeJi");// 锁定单元格不可选中(防止别人直接复制内容到其他excel修改)((SXSSFSheet) writeSheetHolder.getSheet()).lockSelectLockedCells(false);
//        ((SXSSFSheet) writeSheetHolder.getSheet()).lockSelectLockedCells(true);}
}

设置大标题行样式

在这里插入图片描述

@PostMapping("/exportExcel")public void exportExcel(HttpServletResponse response) throws IOException {// 数据库获取数据List<Profile> list = iProfileService.exportExcel(response);List<ProfileExcel> profileExcels = ObjectConvertUtil.convertList(list, ProfileExcel.class);
//        List<ProfileExcel> profileExcels = ExcelConvert.INSTANCE.converList(list);// 输出ExcelUtils.write(response, "大佬信息", "岗位列表", ProfileExcel.class, profileExcels);}

展示工具类中的代码

 /*** 将列表以Excel的形式响应给前端** @param response 响应* @param fileName 文件名* @param sheetName Excel sheet 名* @param head Excel head 头* @param data 数据列表* @param <T> 泛型,保证 head 和 data 类型的一致性* @throws IOException 写入失败的情况*/public static <T> void write(HttpServletResponse response, String fileName, String sheetName, Class<T> head, List<T> data) throws IOException {// 这一部分是设置编码样式response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");response.setCharacterEncoding("utf-8");response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + URLEncoder.encode(fileName, "UTF-8") + ".xlsx");// 输出ExcelEasyExcel.write(response.getOutputStream(),head).relativeHeadRowIndex(2)// 不要自动关闭,交给 Servlet 自己处理//.autoCloseStream(false).registerWriteHandler(new ExportIndexTypeSheetWriteHandler()).registerWriteHandler(new LongestMatchColumnWidthStyleStrategy()).registerWriteHandler(new ExcelFillCellMergeStrategy(3,new int[]{0,1,2,3,4,5,6})).registerWriteHandler(new FontStyleWriteHandler()).sheet(sheetName).doWrite(data);}

核心注册类代码展示

实现SheetWriteHandler策略接口,在sheet创建之前完成对样式进行提前的设置

public class ExportIndexTypeSheetWriteHandler implements SheetWriteHandler {@Overridepublic void afterSheetCreate(SheetWriteHandlerContext context) {// 在创建工作表之后执行的操作WriteSheetHolder writeSheetHolder = context.getWriteSheetHolder();
//        ExcelWriteHeadProperty excelWriteHeadProperty = writeSheetHolder.getExcelWriteHeadProperty();
//        excelWriteHeadProperty.setHeadRowNumber(3);// 设置行高和样式// 创建第一行并设置样式Row row = writeSheetHolder.getSheet().createRow(0);// 设置第一行的高为row.setHeight((short) 1000);// 提前创建第一行单元格并进行设置Cell cell = row.createCell(0);// 设置需要展示的文字信息String title = "大佬们的信息展示\n" +"这个是演示的数据标题\r\n";// 单元格填充内容cell.setCellValue(title);//设置标题// 准备样式// 从上下文内容中获取工作簿并创建样式CellStyle cellStyle = context.getWriteWorkbookHolder().getWorkbook().createCellStyle();// 这个接口在ss。POI的核心模块中存在// 设置单元格的垂直对齐类型 : 垂直对齐以单元格高度为中心。cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);// 设置单元格的水平对齐样式:水平对齐居中,这意味着文本在单元格中居中。cellStyle.setAlignment(HorizontalAlignment.CENTER);// 从上下文获取字体Font font = context.getWriteWorkbookHolder().getWorkbook().createFont();// 设置字体加粗font.setBold(true);// 设置字体行高font.setFontHeight((short) 500);// 设置字体的样式名称font.setFontName("宋体");// 设置字体的颜色font.setColor(HSSFColor.HSSFColorPredefined.RED.getIndex());// 需要在单元格样式中填充字体cellStyle.setFont(font);// 单元格设置为水平居中cellStyle.setAlignment(HorizontalAlignment.CENTER);// 设置单元格为允许换行,不然上述title中的单元格换行无效cellStyle.setWrapText(true);// 最后需要将单元格的样式填充到单元格中cell.setCellStyle(cellStyle);//writeWorkbookHolder.createCellStyle(style,null);// 有点重复操作,从上下文中获取默认第一个sheet页的样式Sheet sheet = context.getWriteSheetHolder().getSheet();// 设置单元格的合并区域,给坐标的方式。哪行开始哪行结束。哪列开始那列结束sheet.addMergedRegionUnsafe(new CellRangeAddress(0, 1, 0, 5));}
}

标题行的样式填充颜色改变

在这里插入图片描述

在这里插入图片描述

颜色9是透明色。

相同单元格的样式合并

在这里插入图片描述

这种编码方式需要在代码使用的时候将那几列放到方法中。

public class ExcelFillCellMergeStrategy implements CellWriteHandler {/*** 合并字段的下标,如第一到五列new int[]{0,1,2,3,4}*/private int[] mergeColumnIndex;/*** 从第几行开始合并,如果表头占两行,这个数字就是2*/private int mergeRowIndex;public ExcelFillCellMergeStrategy() {}public ExcelFillCellMergeStrategy(int mergeRowIndex, int[] mergeColumnIndex) {this.mergeRowIndex = mergeRowIndex;this.mergeColumnIndex = mergeColumnIndex;}@Overridepublic void beforeCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row,Head head, Integer integer, Integer integer1, Boolean aBoolean) {}@Overridepublic void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell,Head head, Integer integer, Boolean aBoolean) {}@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder,List<WriteCellData<?>> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {//当前行int curRowIndex = cell.getRowIndex();//当前列int curColIndex = cell.getColumnIndex();if (curRowIndex > mergeRowIndex) {for (int i = 0; i < mergeColumnIndex.length; i++) {if (curColIndex == mergeColumnIndex[i]) {mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);break;}}}}private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {//获取当前行的当前列的数据和上一行的当前列列数据,通过上一行数据是否相同进行合并Object curData = cell.getCellTypeEnum() == CellType.STRING ? cell.getStringCellValue() :cell.getNumericCellValue();Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);Object preData = preCell.getCellTypeEnum() == CellType.STRING ? preCell.getStringCellValue() :preCell.getNumericCellValue();// 比较当前行的第一列的单元格与上一行是否相同,相同合并当前单元格与上一行//if (curData.equals(preData)) {Sheet sheet = writeSheetHolder.getSheet();List<CellRangeAddress> mergeRegions = sheet.getMergedRegions();boolean isMerged = false;for (int i = 0; i < mergeRegions.size() && !isMerged; i++) {CellRangeAddress cellRangeAddr = mergeRegions.get(i);// 若上一个单元格已经被合并,则先移出原有的合并单元,再重新添加合并单元if (cellRangeAddr.isInRange(curRowIndex - 1, curColIndex)) {sheet.removeMergedRegion(i);cellRangeAddr.setLastRow(curRowIndex);sheet.addMergedRegion(cellRangeAddr);isMerged = true;}}// 若上一个单元格未被合并,则新增合并单元if (!isMerged) {CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, curColIndex,curColIndex);sheet.addMergedRegion(cellRangeAddress);}}}}

单元格的横线设置

在这里插入图片描述

public class FontStyleWriteHandler implements CellWriteHandler {/*** 设置字体样式* @param writeSheetHolder* @param writeTableHolder Nullable.It is null without using table writes.* @param cellDataList     Nullable.It is null in the case of add header.There may be several when fill the data.* @param cell* @param head             Nullable.It is null in the case of fill data and without head.* @param relativeRowIndex Nullable.It is null in the case of fill data.* @param isHead           It will always be false when fill data.*/@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {// 创建字体样式// 设置内容字体样式WriteFont writeFont = new WriteFont();writeFont.setFontName("宋体");for (int i = 0; i < cellDataList.size(); i++) {WriteCellData<?> writeCellData = cellDataList.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();// 横线的设置writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setWriteFont(writeFont);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setShrinkToFit(true);}}}

导出字体的样式自定义

在这里插入图片描述

public class FontStyleWriteHandler implements CellWriteHandler {/*** 设置字体样式* @param writeSheetHolder* @param writeTableHolder Nullable.It is null without using table writes.* @param cellDataList     Nullable.It is null in the case of add header.There may be several when fill the data.* @param cell* @param head             Nullable.It is null in the case of fill data and without head.* @param relativeRowIndex Nullable.It is null in the case of fill data.* @param isHead           It will always be false when fill data.*/@Overridepublic void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {// 创建字体样式// 设置内容字体样式WriteFont writeFont = new WriteFont();writeFont.setFontName("宋体");// 核心设置字体样式writeFont.setColor(IndexedColors.RED.getIndex());for (int i = 0; i < cellDataList.size(); i++) {WriteCellData<?> writeCellData = cellDataList.get(i);WriteCellStyle writeCellStyle = writeCellData.getWriteCellStyle();writeCellStyle.setBorderLeft(THIN);writeCellStyle.setBorderRight(THIN);writeCellStyle.setBorderTop(THIN);writeCellStyle.setBorderBottom(THIN);writeCellStyle.setWriteFont(writeFont);writeCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);writeCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);writeCellStyle.setShrinkToFit(true);}}}

多级标题实现

在这里插入图片描述

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/70858.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

实现无公网IP的公网环境下Windows远程桌面Ubuntu 18.04连接,高效远程办公!

文章目录 一、 同个局域网内远程桌面Ubuntu1. 更新软件仓库2. 安装支持包3. 安装XFCE4桌面环境4. 安装XRDP5. 环境设置5.1 XFCE桌面配置5.2 在配置文件中&#xff0c;加入XFCE会话 6 重启服务7. 查看IP地址8. 使用Windows远程桌面连接 二、公网环境系统远程桌面Ubuntu1. 注册cp…

上架亚马逊CPC证书常用测试标准总结,CPSC总共公布了38个标准

在亚马逊美国站&#xff0c;上架与儿童相关的产品&#xff0c;包括玩具、婴童用品等&#xff0c;会被要求出CPC证书&#xff0c;否则将面临产品被下架、货款被扣留、被起诉等风险&#xff01;CPC认证 CPC认证就是儿童产品安全证书&#xff08;Children’s Product Certificate,…

前端js读取本地md或txt文件内容

<!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport" content"widthdevice-width, initial-scale1.0" /><title>前端js读取本地md或txt文件内容</title>&l…

面试系列 - Java代理模式详解

代理模式是一种结构型设计模式&#xff0c;它允许你提供一个代理或替代品来控制对另一个对象的访问。代理对象通常充当客户端和实际目标对象之间的中介&#xff0c;可以用于各种用途&#xff0c;例如延迟加载、访问控制、监视、记录日志等。 一、代理模式包括以下几个关键角色…

嵌入式BL304可提供4路RS485/RS232口

BL304网关具有多功能的数据通信和处理设备&#xff0c;具有多种输入和输出接口&#xff0c;支持多种操作系统和应用程序&#xff0c;以及数据库连接。下面我们将详细介绍该产品的特点和功能。 设备提供了多路RS485/RS232接口、CAN口、网口、DI口、PWM口输出以及USB接口&#xf…

肖sir__设计测试用例方法之状态迁移法05_(黑盒测试)

设计测试用例方法之状态迁移法 一、状态迁移图 定义&#xff1a;通过描绘系统的状态及引起系统状态转换的事件&#xff0c;来表示系统的行为 案例&#xff1a; &#xff08;1&#xff09; 订机票案例1&#xff1a; l向航空公司打电话预定机票—>此时机票信息处于“完成”状…

API签名认证的说明及实现

目录 请思考一个问题什么是API签名认证为什么需要API签名认证如何在后端实现签名认证签名认证实现通过 http request header 头传递参数加密方式怎么知道这个签名对不对&#xff1f;怎么防重放&#xff1f; Go 代码实现sign.goservice.goclient.go 请思考一个问题 请思考一个重…

【pytorch】数据加载dataset和dataloader的使用

1、dataset加载数据集 dataset_tranform torchvision.transforms.Compose([torchvision.transforms.ToTensor(),])train_set torchvision.datasets.CIFAR10(root"./train_dataset",trainTrue,transformdataset_tranform,downloadTrue) test_set torchvision.data…

企业网络安全:威胁检测和响应 (TDR)

什么是威胁检测和响应 威胁检测和响应&#xff08;TDR&#xff09;是指识别和消除 IT 基础架构中存在的恶意威胁的过程。它涉及主动监控、分析和操作&#xff0c;以降低风险并防止未经授权的访问、恶意活动和数据泄露&#xff0c;以免它们对组织的网络造成任何潜在损害。威胁检…

Origin绘制彩色光谱图

成果图 1、双击线条打开如下窗口 2、选择“图案”-》颜色-》按点-》映射-》Wavelength 3、选择颜色映射 4、单击填充-》选择加载调色板-》Rainbow-》确定 5、单击级别&#xff0c;设置成从370到780&#xff0c;右侧增量选择2&#xff08;越小&#xff0c;颜色渐变越细腻&am…

linux之perf(2)list事件

Linux之perf(2)list事件 Author&#xff1a;Onceday Date&#xff1a;2023年9月3日 漫漫长路&#xff0c;才刚刚开始… 参考文档: Tutorial - Perf Wiki (kernel.org)perf-list(1) - Linux manual page (man7.org) 1. 概述 perf list用于列出可用的性能事件&#xff0c;这…

Elasticsearch:利用向量搜索进行音乐信息检索

作者&#xff1a;Alex Salgado 欢迎来到音乐信息检索的未来&#xff0c;机器学习、向量数据库和音频数据分析融合在一起&#xff0c;带来令人兴奋的新可能性&#xff01; 如果你对音乐数据分析领域感兴趣&#xff0c;或者只是热衷于技术如何彻底改变音乐行业&#xff0c;那么本…

oracle10和11功能说明比较

Oracle 10g/11g的特点和优势 首先&#xff0c;Oracle 10g/11g具有以下几个特点&#xff1a; 1. 可靠性和稳定性&#xff1a;Oracle 10g采用了多种技术来确保数据的可靠性和稳定性&#xff0c;如ACID事务处理和数据备份与恢复机制。它还提供了高可用性的解决方案&#xff0c;如…

记本地新建一个gradle方式springboot项目过程

打算使用gradle在idea新建个springboot项目&#xff0c;然后坑很多&#xff0c;记录一下 原来我的idea应该是社区版&#xff0c;新建项目时候没有可以选择spring相关配置&#xff0c;然后卸载了重装&#xff0c;之前问题是启动是启动起来了&#xff0c;但是状态栏那边一直显示…

手写Spring:第8章-初始化和销毁方法

文章目录 一、目标&#xff1a;初始化和销毁方法二、设计&#xff1a;初始化和销毁方法三、实现&#xff1a;初始化和销毁方法3.1 工程结构3.2 Spring应用上下文和Bean对象扩展初始化和销毁类图3.3 定义初始化和销毁方法的接口3.3.1 定义初始化接口3.3.2 定义销毁接口3.3.3 定义…

机器学习基础算法--回归类型和评价分析

目录 1.数据归一化处理 2.数据标准化处理 3.Lasso回归模型 4.岭回归模型 5.评价指标计算 1.数据归一化处理 """ x的归一化的方法还是比较多的我们就选取最为基本的归一化方法 x(x-x_min)/(x_max-x_min) """ import numpy as np from sklea…

CSS构建基础(二)选择器

在CSS中&#xff0c;选择器用于定位我们想要样式化的网页上的HTML元素。有各种各样的CSS选择器可用&#xff0c;允许在选择要样式化的元素时实现细粒度的精度。在本文及其子文章中&#xff0c;我们将详细介绍不同的类型&#xff0c;了解它们是如何工作的。 1、什么是选择器? …

浏览器跨域

相关问题 什么是跨域为什么会跨域为什么会有跨域的限制怎么解决跨域 回答关键点 CORS和同源策略 跨域问题的来源是浏览器为了请求安全而引入的基于同源策略的安全特性。当页面和请求的协议、主机名或端口不同时&#xff0c;浏览器判定两者不同源&#xff0c;即为跨域请求。需…

Springboot开发时,对前端的请求参数,后端用于接受的实体类有没有必要校验为null?

首先给出结论&#xff1a;不用校验为NULL,如果null,Springboot会直接抛异常而不是返回NUll。只需要对其中的属性判断是否null 问题代码如下&#xff1a; public R<Boolean> addzbsz (RequestBody RequestzbszAdd requestzbszAdd) {if ( requestzbszAddnull){return true…

PMP备考全攻略,看这一份就够了!

免费送备考资料。 Project Management Professional (PMP) - 项目管理专业人士,简称PMP。是项目管理专业人士资格认证由项目管理学会和 PMI 发起&#xff0c;严格评估管理项目人员知识技能是否具有高品质的资格认证考试。 简单来说&#xff0c;PMP是美国PMI发起的评估项目管理…