使用POI导出Excel(二)-利用模板

一、基本操作见: 使用POI导出Excel

二、本次功能需求

  给了一个模板,里面有6个sheet页,每页里面都需要填充相应的数据。如图:

三、需求分析

  1、分了6个sheet页,每页的数据都不一样,首先代码里要获得它们的数据,然后6个sheet页只能一个个填进去,没法批量操作。

  2、如果直接代码创建Excel并画表格样式和数据,那么工程量将会很大,而且会导致代码很乱。我采用的方法是把模板保存到项目里,再通过路径获取到该模板文件,把其内容全部复制到新创建的Excel中,再往里填充数据。

  3、注意到第一张图中有总计这一行,我的建议是先在代码中计算出相应的数据,再填充,我也是这样做的。当然,我的第一想法是给它设置函数公式,但是后来我并没有这么做。

  4、从第二张图中可以看得出要填充的数据的行是不固定的,除非规定了只填充前几条数据。而在11、12行中的数据是已经有了的,当填充的数据太多,就会覆盖掉那行数据。有两种方法解决,一是把它获取到,先填充数据覆盖掉,再在最后一条数据的下一行把它再填充回去。二是根据数据的总条数通过代码往中间插入空白行。两种方法都可行,我采用了第二种。

四、上代码

 1     public void exportAll() {
 2         OutputStream out = null;
 3         try {
 4             // 获取数据
 5             List<BaseTalentFlowAnalysisGridVo> talentFlows = this.talentFlowQuestionnaireService.listAnalysis();
 6             List<BaseBasicSituationAnalysisGridVo> basicSituations = this.basicSituationService.listAnalysis();
 7             List<BaseTalentInputAnalysisGridVo> talentInputs = this.talentInputService.listAnalysis();
 8             List<BaseTalentDemandAnalysisGridVo> talentDemands = this.talentDemandService.listAnalysis();
 9             List<BaseHighLevelTalentsRosterAnalysisGridVo> talentRosters = this.rosterService.listAnalysis();
10             Map<String, Object> results = new HashMap<>();
11             results.put("talentFlows", talentFlows);
12             results.put("basicSituations", basicSituations);
13             results.put("talentInputs", talentInputs);
14             results.put("talentDemands", talentDemands);
15             results.put("talentRosters", talentRosters);
16 
17             // 拿到模板文件
18             String path = ServletActionContext.getServletContext().getRealPath("/");
19             String filePath = path + "\\resources\\temp\\人才统计报表模板.xls";
20             FileInputStream tps = new FileInputStream(new File(filePath));
21             final HSSFWorkbook tpWorkbook = new HSSFWorkbook(tps);
22             out = response.getOutputStream();
23             response.reset();
24             response.setHeader("content-disposition",
25                     "attachment;filename=" + new String(("人才统计报表").getBytes("gb2312"), "ISO8859-1") + ".xls");
26             response.setContentType("APPLICATION/msexcel");
27             // 新建一个Excel的工作空间
28             HSSFWorkbook workbook = new HSSFWorkbook();
29             // 把模板复制到新建的Excel
30             workbook = tpWorkbook;
31             // 填充数据
32             this.excelService.addData(workbook, results);
33             // 输出Excel内容,生成Excel文件
34             workbook.write(out);
35         } catch (final IOException e) {
36             LOGGER.error(e);
37         } catch (final IllegalArgumentException e) {
38             LOGGER.error(e);
39         } catch (final Exception e) {
40             LOGGER.error(e);
41         } finally {
42             try {
43                 // 最后记得关闭输出流
44                 response.flushBuffer();
45                 if (out != null) {
46                     out.flush();
47                     out.close();
48                 }
49             } catch (final IOException e) {
50                 LOGGER.error(e);
51             }
52         }
53     }
 1     /**
 2      * 第二个sheet页:人才流动情况表
 3      * 
 4      * @param workbook
 5      * @param talentFlows
 6      */
 7     private void addTalentFlows(HSSFWorkbook workbook, List<BaseTalentFlowAnalysisGridVo> talentFlows) {
 8         // 获取第二个sheet页
 9         Sheet talentFlowSheet = workbook.getSheetAt(1);
10         Row talentFlowRow = talentFlowSheet.getRow(0);
11         // 声明总计的那几个数据
12         Integer totalLastYearTotal = 0;
13         Integer totalHighLevelTalent = 0;
14         Integer totalUndergraduateAndGraduate = 0;
15         Integer totalCollegeStudents = 0;
16         Integer totalSocialTalent = 0;
17         Integer totalMilitaryTransferCadres = 0;
18         Integer totalReturnees = 0;
19         Integer totalRetirement = 0;
20         Integer totalResignation = 0;
21         Integer totalDismiss = 0;
22         Integer totalOther = 0;
23         Integer totalAverageIncrease = 0;
24         Integer totalAverageReduction = 0;
25         // 循环数据
26         for (BaseTalentFlowAnalysisGridVo baseTalentFlowAnalysisGridVo : talentFlows) {
27             // 循环行
28             for (int tr = 4; tr < 11; tr++) {
29                 talentFlowRow = talentFlowSheet.getRow(tr);
30                 // 当数据的项目字段和行的第一列内容相等,则把该条数据填入该行
31                 if (talentFlowRow.getCell(0).getStringCellValue()
32                         .equals(baseTalentFlowAnalysisGridVo.getFkProjectName())) {
33                     talentFlowRow.getCell(1).setCellValue(baseTalentFlowAnalysisGridVo.getLastYearTotal());
34                     // 给总计做计算
35                     totalLastYearTotal += baseTalentFlowAnalysisGridVo.getLastYearTotal();
36                     talentFlowRow.getCell(2).setCellValue(baseTalentFlowAnalysisGridVo.getHighLevelTalent());
37                     totalHighLevelTalent += baseTalentFlowAnalysisGridVo.getHighLevelTalent();
38                     talentFlowRow.getCell(3).setCellValue(baseTalentFlowAnalysisGridVo.getUndergraduateAndGraduate());
39                     totalUndergraduateAndGraduate += baseTalentFlowAnalysisGridVo.getUndergraduateAndGraduate();
40                     talentFlowRow.getCell(4).setCellValue(baseTalentFlowAnalysisGridVo.getCollegeStudents());
41                     totalCollegeStudents += baseTalentFlowAnalysisGridVo.getCollegeStudents();
42                     talentFlowRow.getCell(5).setCellValue(baseTalentFlowAnalysisGridVo.getSocialTalent());
43                     totalSocialTalent += baseTalentFlowAnalysisGridVo.getSocialTalent();
44                     talentFlowRow.getCell(6).setCellValue(baseTalentFlowAnalysisGridVo.getMilitaryTransferCadres());
45                     totalMilitaryTransferCadres += baseTalentFlowAnalysisGridVo.getMilitaryTransferCadres();
46                     talentFlowRow.getCell(7).setCellValue(baseTalentFlowAnalysisGridVo.getReturnees());
47                     totalReturnees += baseTalentFlowAnalysisGridVo.getReturnees();
48                     talentFlowRow.getCell(8).setCellValue(baseTalentFlowAnalysisGridVo.getRetirement());
49                     totalRetirement += baseTalentFlowAnalysisGridVo.getRetirement();
50                     talentFlowRow.getCell(9).setCellValue(baseTalentFlowAnalysisGridVo.getResignation());
51                     totalResignation += baseTalentFlowAnalysisGridVo.getResignation();
52                     talentFlowRow.getCell(10).setCellValue(baseTalentFlowAnalysisGridVo.getDismiss());
53                     totalDismiss += baseTalentFlowAnalysisGridVo.getDismiss();
54                     talentFlowRow.getCell(11).setCellValue(baseTalentFlowAnalysisGridVo.getOther());
55                     totalOther += baseTalentFlowAnalysisGridVo.getOther();
56                     talentFlowRow.getCell(12).setCellValue(baseTalentFlowAnalysisGridVo.getAverageIncrease());
57                     totalAverageIncrease += baseTalentFlowAnalysisGridVo.getAverageIncrease();
58                     talentFlowRow.getCell(13).setCellValue(baseTalentFlowAnalysisGridVo.getAverageReduction());
59                     totalAverageReduction += baseTalentFlowAnalysisGridVo.getAverageReduction();
60                     break;
61                 }
62             }
63         }
64         // 给总计行填充数据
65         talentFlowRow = talentFlowSheet.getRow(11);
66         talentFlowRow.getCell(1).setCellValue(totalLastYearTotal);
67         talentFlowRow.getCell(2).setCellValue(totalHighLevelTalent);
68         talentFlowRow.getCell(3).setCellValue(totalUndergraduateAndGraduate);
69         talentFlowRow.getCell(4).setCellValue(totalCollegeStudents);
70         talentFlowRow.getCell(5).setCellValue(totalSocialTalent);
71         talentFlowRow.getCell(6).setCellValue(totalMilitaryTransferCadres);
72         talentFlowRow.getCell(7).setCellValue(totalReturnees);
73         talentFlowRow.getCell(8).setCellValue(totalRetirement);
74         talentFlowRow.getCell(9).setCellValue(totalResignation);
75         talentFlowRow.getCell(10).setCellValue(totalDismiss);
76         talentFlowRow.getCell(11).setCellValue(totalOther);
77         talentFlowRow.getCell(12).setCellValue(totalAverageIncrease);
78         talentFlowRow.getCell(13).setCellValue(totalAverageReduction);
79     }
 1     /**
 2      * 第四个sheet页:人才需求情况调查表
 3      * 
 4      * @param workbook
 5      * @param talentFlows
 6      */
 7     private void addalentDemands(HSSFWorkbook workbook, List<BaseTalentDemandAnalysisGridVo> talentDemands)
 8             throws IllegalArgumentException, IllegalAccessException {
 9         Sheet talentDemandSheet = workbook.getSheetAt(3);
10         Row talentDemandRow = talentDemandSheet.getRow(4);
11         // 如果数据大于模板中的行数,插入行并复制第一行数据的格式
12         if (talentDemands.size() > 5) {
13             // 插入行,5是模板中已有的行数
14             talentDemandSheet.shiftRows(5, talentDemandSheet.getLastRowNum(), talentDemands.size() - 5, true, false);
15             Row sourceRow = talentDemandSheet.getRow(4);
16             for (int i = 0; i < talentDemands.size() - 5; i++) {
17                 Row newRow = talentDemandSheet.createRow(4 + i + 1);
18                 newRow.setHeight(sourceRow.getHeight());
19                 for (int j = 0; j < sourceRow.getLastCellNum(); j++) {
20                     Cell templateCell = sourceRow.getCell(j);
21                     if (templateCell != null) {
22                         Cell newCell = newRow.createCell(j);
23                         copyCell(templateCell, newCell);
24                     }
25                 }
26             }
27         }
28         // 填充数据
29         for (int i = 0; i < talentDemands.size(); i++) {
30             talentDemandRow = talentDemandSheet.getRow(4 + i);
31             talentDemandRow.getCell(0).setCellValue(talentDemands.get(i).getPositionTitle());
32             talentDemandRow.getCell(2).setCellValue(talentDemands.get(i).getDemand());
33             talentDemandRow.getCell(3).setCellValue(talentDemands.get(i).getAge());
34             talentDemandRow.getCell(4).setCellValue(talentDemands.get(i).getFkAcademicDegreeName());
35             talentDemandRow.getCell(5).setCellValue(talentDemands.get(i).getTechnicalTitles());
36             talentDemandRow.getCell(6).setCellValue(talentDemands.get(i).getProfession());
37             talentDemandRow.getCell(7).setCellValue(talentDemands.get(i).getFkTalentCategoryName());
38             talentDemandRow.getCell(8).setCellValue(talentDemands.get(i).getFkServiceFormName());
39             talentDemandRow.getCell(9).setCellValue(talentDemands.get(i).getProvide());
40             talentDemandRow.getCell(10).setCellValue(talentDemands.get(i).getOtherCases());
41             talentDemandRow.getCell(11).setCellValue(talentDemands.get(i).getFkIntentionToChooseName());
42         }
43     }

五、缺点:一手烂代码,应该给对象设置对应的中文注释,和模板中的列头一样,然后再通过循环填充数据。这样一个个填上去太傻了。

六、通过实验测试,在getCell的时候,如果那是个合并的单元格,那么该单元格的数据存在了左上角,其他的格内容为空。

猜你喜欢

转载自www.cnblogs.com/qq765065332/p/9221015.html