POI生成EXCEL文件

一、背景

根据指定格式的JSON文件生成对应的excel文件,需求如下

  • 支持多sheet
  • 支持单元格合并
  • 支持插入图片
  • 支持单元格样式可定制
  • 需要 标题(title),表头(head),数据(data) ,表尾(foot) 明确区分

二、效果预览

三、数据格式

由于是生成Excel文件,这里值考虑生成xlsx格式的Excel文件,数据多表头默认考虑使用 | 表示,不在使用colspan rowspan作为。如需要表示两列两行,第一列合并表头格式为: A|B,A|C生成的表格为

A
B C

前端通过post的方式将需要生成的数据构造成符合要求的JSON文件提交跟后台。根据以上需求定义JSON格式如下

  1. {
  2. "saveName": "生成Excel的文件名.xlsx",
  3. "userStyles": [{
  4. "id": "1", //不能出现重复,在需要设置单元样式的地方,可以直接将style赋值为此值
  5. "style": {
  6. "font": { //设置字体基本格式
  7. "blod": true,//是否加粗
  8. "italic": true, //是否倾斜
  9. "color": "#FF0000",//字体颜色
  10. "name": "微软雅黑", //字体名称
  11. "height": 20 //大小
  12. },
  13. "fmtStr": "", //单元格格式,#,##0.00_);#,##0.00;0 千分位
  14. "align": "",//水平对齐方式 left right center
  15. "valign": "",//垂直对齐方式 top center bottom
  16. "borderColor": "", //设置边框颜色 如 #FF0000
  17. "bgColor": "" //设置单元格填充颜色
  18. }
  19. }],
  20. "sheets": [{
  21. "sheetName": "", //sheet名称
  22. "title": [], // 对应Sheet标题区域数据
  23. "titleMerge": [], //对应Sheet标题区域合并信息
  24. "head": [{}], //表头信息
  25. "data": [], //数据信息
  26. "dataMerge": [], //数据合并信息
  27. "foot": [], //表尾信息
  28. "footMerge": [], //表尾合并信息
  29. "img": [] //图片信息,需要将图片转换base64
  30. }]
  31. }

简要说明

  • head 数组中为JSON对象格式为
  1. {
  2. "name": "A|B", //表头名称,多表头用|分割
  3. "type": "str", //此列数据类型 str num ,在excel中日期也是数字类型,通过fmtStr,显示为日期格式
  4. "field": "F_FIELD1", //备用字段,可不用
  5. "style": { //此列数据为列默认样式,可以是Style对象,也可以是在userStyles中定义的id值
  6. "align": "center"
  7. }
  8. }
  • 在数组 title data foot 中,列表中的数据,可以是一个单独的值如 1,”a”,也可以是一个对象,当为对象时,格式为
  1. {
  2. "value": "", //单元格具体的值
  3. "type": "", //单元格类型,默认str
  4. "style": {} //单元格样式 可以是Style对象,也可以是在userStyles中定义的id值,如果没设置,默认取head总此列对应的style
  5. }
  • titleMerge、dataMerge、footMerge数组值为逗号分隔的字符串,其含义为 "开始行,结束行,开始列,结束列",索引从0开始。如在title中有两行三列数据,现在需要合并一行两列数据对应的值为"0,0,0,1"
  • img数组中值为对象,格式
  1. {
  2. "col": 1, //图片开始列
  3. "row": 0, //开始行
  4. "colSpan": 1,//列跨度,最小值1
  5. "rowSpan": 2, //行跨度,最小值1
  6. "data": "" //base64图片数据如: "...ggg=="
  7. }

四、关键实现

07以后的Excle文件,其实是一个压缩包,里边是一个个的xml文件,其中每一个sheet是一个xml文件,样式是一个xml文件,图片是对应的图片文件,放在media文件夹中,所以,代码思路依次为

  • 构建 XSSFWorkbook 对象
  • 生成样式
  • 依次生成,title head data foot 行数据
  • 依次处理合并信息 titlemerge datamerge footmerge
  • 添加图片信息
  • 输出文件流

功能入口如下

  1. @Override
  2. public void buildOutputStream() throws FileProducerException {
  3. // 处理传入的JSON数据
  4. sheets = this.jsonData.getJSONArray(this.SHEETS);
  5. Iterator<Object> sheetIter = sheets.iterator();
  6. if (sheets.isEmpty()) {
  7. this.responseData.setErrcode(1001);
  8. this.responseData.setSuccess(false);
  9. this.responseData.setErrmsg("无数据可生成");
  10. throw new FileProducerException();
  11. }
  12. wb = new XSSFWorkbook();
  13. // 建立全局格式
  14. JSONArray userStyles = this.jsonData.getJSONArray(this.USERSTYLES);
  15. this.initUserStyles(userStyles);
  16. this.initDefaultHeadStyle();
  17.  
  18. XSSFSheet ws;
  19. JSONObject sheet;
  20. JSONArray sheetData;
  21. JSONArray sheetTitle;
  22. JSONArray sheetHead;
  23. JSONArray sheetFoot;
  24. JSONArray sheetImgs;
  25.  
  26. String sheetName;
  27. int sheetIndex = 0;
  28. while (sheetIter.hasNext()) {
  29. sheet = (JSONObject) sheetIter.next();
  30. // 获取sheet名称
  31. sheetName = sheet.getString(this.SHEET_NAME);
  32. ws = wb.createSheet();
  33. if (StringUtils.isNotBlank(sheetName)) {
  34. wb.setSheetName(sheetIndex, sheetName);
  35. }
  36. int sheetRowIndex = 0;
  37. sheetTitle = sheet.getJSONArray(this.SHEET_TITLE);
  38. this.setMergeCells(ws, sheet.getJSONArray(this.SHEET_TITLE_MERGE),
  39. sheetRowIndex);
  40. sheetRowIndex = this.createRandom(ws, sheetTitle, sheetRowIndex);
  41.  
  42. sheetHead = sheet.getJSONArray(this.SHEET_HEAD);
  43. sheetRowIndex = this.createHeadColumn(ws, sheetHead, sheetRowIndex);
  44.  
  45. this.setMergeCells(ws, sheet.getJSONArray(this.SHEET_DATA_MERGE),
  46. sheetRowIndex);
  47. sheetData = sheet.getJSONArray(this.SHEET_DATA);
  48. sheetRowIndex = this.createData(ws, sheetData, sheetRowIndex);
  49.  
  50. sheetFoot = sheet.getJSONArray(this.SHEET_FOOT);
  51. this.setMergeCells(ws, sheet.getJSONArray(this.SHEET_FOOT_MERGE),
  52. sheetRowIndex);
  53. sheetRowIndex = this.createRandom(ws, sheetFoot, sheetRowIndex);
  54.  
  55. sheetImgs = sheet.getJSONArray(this.SHEET_IMG);
  56.  
  57. this.setSheetImages(ws, sheetImgs);
  58. }
  59.  
  60. // 返回输出流
  61. try {
  62. ByteArrayOutputStream os = new ByteArrayOutputStream();
  63. wb.write(os);
  64. this.outStreams.add(os);
  65. } catch (IOException e) {
  66. throw new FileProducerException(e.getMessage(), e.getCause());
  67. }
  68. }

生成单元格样式对象,包括字体 边框 背景 对齐方式

  1. private XSSFCellStyle createCellStyle(JSONObject style) {
  2.  
  3. XSSFCellStyle cellStyle = wb.createCellStyle();
  4. // 设置字体
  5. JSONObject font = style.getJSONObject(this.STYLE_FONT);
  6. Font excelFont = this.createFont(font);
  7. if (excelFont != null) {
  8. cellStyle.setFont(excelFont);
  9. }
  10. // border统一黑色
  11. cellStyle.setBorderBottom(BorderStyle.THIN);
  12. cellStyle.setBorderTop(BorderStyle.THIN);
  13. cellStyle.setBorderLeft(BorderStyle.THIN);
  14. cellStyle.setBorderRight(BorderStyle.THIN);
  15.  
  16. String borderColor = style.getString(this.BORDER_COLOR);
  17. if (StringUtils.isNotBlank(borderColor)) {
  18. XSSFColor xfBorderColor = new XSSFColor(new Color(Integer.parseInt(
  19. borderColor.substring(1), 16)));
  20. cellStyle.setBorderColor(BorderSide.BOTTOM, xfBorderColor);
  21. cellStyle.setBorderColor(BorderSide.TOP, xfBorderColor);
  22. cellStyle.setBorderColor(BorderSide.LEFT, xfBorderColor);
  23. cellStyle.setBorderColor(BorderSide.RIGHT, xfBorderColor);
  24. }
  25. // 背景色
  26. String bgColor = style.getString(this.BACKGROUND_COLOR);
  27. if (StringUtils.isNotBlank(bgColor)) {
  28. XSSFColor cellBgColor = new XSSFColor(new Color(Integer.parseInt(
  29. bgColor.substring(1), 16)));
  30. cellStyle.setFillForegroundColor(cellBgColor);
  31. cellStyle.setFillPattern(FillPatternType.SOLID_FOREGROUND);
  32. }
  33. // 对齐方式
  34. String hAlignment = style.getString(this.HALIGNMENT);
  35. if (StringUtils.isNotBlank(hAlignment))
  36. cellStyle.setAlignment(HorizontalAlignment.valueOf(hAlignment
  37. .toUpperCase()));
  38. String vAlignment = style.getString(this.VALIGNMENT);
  39. if (StringUtils.isNotBlank(vAlignment))
  40. cellStyle.setVerticalAlignment(VerticalAlignment.valueOf(vAlignment
  41. .toUpperCase()));
  42. // 自动换行TRUE
  43. cellStyle.setWrapText(true);
  44.  
  45. // 格式
  46. String fmt = style.getString(this.FMTSTRING);
  47. if (StringUtils.isNotBlank(fmt))
  48. cellStyle.setDataFormat(wb.createDataFormat().getFormat(fmt));
  49. return cellStyle;
  50. }

创建字体样式

  1. private Font createFont(JSONObject fontCfg) {
  2. if (fontCfg == null)
  3. return null;
  4. XSSFFont font = wb.createFont();
  5. font.setFontName(fontCfg.getString(this.FONT_NAME));
  6. Boolean fontBoole = fontCfg.getBoolean(FONT_BLOD);
  7. if (fontBoole != null)
  8. font.setBold(fontBoole.booleanValue());
  9. fontBoole = fontCfg.getBoolean(this.FONT_ITALIC);
  10. if (fontBoole != null)
  11. font.setItalic(fontBoole.booleanValue());
  12. fontBoole = fontCfg.getBoolean(this.FONT_UNDERLINE);
  13. if (fontBoole != null && fontBoole.booleanValue() == true)
  14. font.setUnderline(FontUnderline.SINGLE.getByteValue());
  15. Short fontHeight = fontCfg.getShort(this.FONT_HEIGHT);
  16. if (fontHeight != null)
  17. font.setFontHeightInPoints(fontHeight);
  18. String colorStr = fontCfg.getString(this.FONT_COLOR);
  19. if (colorStr != null) {
  20. font.setColor(new XSSFColor(new Color(Integer.parseInt(
  21. colorStr.substring(1), 16))));
  22. }
  23. return font;
  24. }

处理表头,表过多表头处理,采用 | 分割的方式,传入head长度为列数据,name中有几个 | 就知道表头有几行。所以针对表头处理有以下几个步骤

  • 生成默认列样式
  • 填充所有列数据,求出最大行数
  • 横向合并内容相同的单元
  • 纵向合并空白的单元格
  1. private int createHeadColumn(XSSFSheet ws, JSONArray sheetHead,
  2. int sheetRowIndex) {
  3. if (sheetHead == null)
  4. return sheetRowIndex;
  5. Iterator<Object> headIter = sheetHead.iterator();
  6. JSONObject curHead = null;
  7. int colIndex = 0;
  8. Object colStyle = null;
  9. int colSize = sheetHead.size();
  10. headTypes = new String[colSize];
  11. headCellStyleKeys = new String[colSize];
  12. int[] headColLevel = new int[colSize];
  13. String colName = null;
  14. String[] colNameAry = null;
  15. int maxLevel = 0;
  16. int colLevel = 0;
  17. XSSFCell headCell = null;
  18. ArrayList<ArrayList<String>> headValueList = new ArrayList<ArrayList<String>>();
  19. while (headIter.hasNext()) {
  20. curHead = (JSONObject) headIter.next();
  21. // 处理默认样式
  22. if (curHead.containsKey(this.COLUMN_STYLE)) {
  23. colStyle = curHead.get(this.COLUMN_STYLE);
  24. if (colStyle instanceof JSONObject) {
  25. headCellStyleKeys[colIndex] = this.COLUMNSTYLE_PREV
  26. + colIndex;
  27. this.userStyles.put(headCellStyleKeys[colIndex],
  28. this.createCellStyle((JSONObject) colStyle));
  29. } else if (this.userStyles.containsKey(colStyle)) {
  30. headCellStyleKeys[colIndex] = (String) colStyle;
  31. }
  32. }
  33. // 处理默认列宽
  34. if (curHead.containsKey(this.COLUMN_WIDTH)) {
  35. ws.setDefaultColumnWidth(pixToExcelWdith(curHead
  36. .getIntValue(this.COLUMN_WIDTH)));
  37. }
  38. // 保存列样式
  39. if (curHead.containsKey(this.COLUMN_TYPE)) {
  40. headTypes[colIndex] = curHead.getString(this.COLUMN_TYPE);
  41. } else {
  42. headTypes[colIndex] = this.CELLTYPESTRING;
  43. }
  44. // 处理多表头
  45. colName = curHead.getString(this.COLUMN_NAME);
  46. colNameAry = colName.split("\\|");
  47. colLevel = colNameAry.length;
  48. headColLevel[colIndex] = colLevel;
  49. if (colLevel > maxLevel) {
  50. maxLevel = colLevel;
  51. }
  52. for (int i = 0; i < colLevel; i++) {
  53. if (headValueList.size() <= i) {
  54. headValueList.add(new ArrayList<String>());
  55. }
  56. headValueList.get(i).add(colIndex, colNameAry[i]);
  57. XSSFRow row = ws.getRow(sheetRowIndex + i);
  58. if (row == null) {
  59. row = ws.createRow(sheetRowIndex + i);
  60. }
  61. headCell = row.createCell(colIndex);
  62. headCell.setCellValue(colNameAry[i]);
  63. headCell.setCellStyle(this.userStyles.get(this.HEADSTYLE_KEY));
  64. }
  65. colIndex++;
  66. }
  67.  
  68. // 横向合并
  69. Iterator<ArrayList<String>> a = headValueList.iterator();
  70. JSONArray headMerge = new JSONArray();
  71. String prev = "";
  72. String curent = null;
  73. int lRowIndex = 0;
  74. int startCol = 0;
  75. int mergeCol = 0;
  76. ArrayList<String> columnInfo = null;
  77. while (a.hasNext()) {
  78. startCol = 0;
  79. mergeCol = 0;
  80. prev = "";
  81. columnInfo = a.next();
  82. // 第三列才能知道,第一列和第二列是否合并
  83. columnInfo.add("");
  84. Iterator<String> b = columnInfo.iterator();
  85. XSSFCell lastRowCell = null;
  86. while (b.hasNext()) {
  87. curent = b.next();
  88. if (lRowIndex > 0) {
  89. lastRowCell = ws.getRow(sheetRowIndex + lRowIndex - 1)
  90. .getCell(startCol);
  91. }
  92. if (prev.equalsIgnoreCase(curent) && lRowIndex == 0) {
  93. ws.getRow(sheetRowIndex + lRowIndex).getCell(startCol)
  94. .setCellType(Cell.CELL_TYPE_BLANK);
  95. mergeCol++;
  96. } else if (prev.equalsIgnoreCase(curent)
  97. && lRowIndex > 0
  98. && StringUtils
  99. .isBlank(lastRowCell.getStringCellValue())) {
  100. ws.getRow(sheetRowIndex + lRowIndex).getCell(startCol)
  101. .setCellType(Cell.CELL_TYPE_BLANK);
  102. mergeCol++;
  103. } else {
  104. if (mergeCol > 0 && startCol > 0) {
  105. headMerge.add(String.format("%d,%d,%d,%d", lRowIndex,
  106. lRowIndex, startCol - mergeCol - 1,
  107. startCol - 1));
  108. mergeCol = 0;
  109. }
  110. }
  111. startCol++;
  112. prev = curent;
  113. }
  114. lRowIndex++;
  115. }
  116. for (int i = 0; i < colSize; i++) {
  117. if (headColLevel[i] < maxLevel) { // 存在列合并
  118. headMerge.add(String.format("%d,%d,%d,%d", headColLevel[i] - 1,
  119. maxLevel - 1, i, i));
  120. for (int r = headColLevel[i]; r < maxLevel; r++) {
  121. ws.getRow(sheetRowIndex + r)
  122. .createCell(i)
  123. .setCellStyle(
  124. this.userStyles.get(this.HEADSTYLE_KEY));
  125. }
  126. }
  127. }
  128.  
  129. this.setMergeCells(ws, headMerge, sheetRowIndex);
  130. return sheetRowIndex + maxLevel;
  131. }

添加图片,默认采用单元格描点方式,将图片固定指定的单元格区域内

  1. private void addImg(XSSFSheet ws, JSONObject img, XSSFCreationHelper cHelper) {
  2. String imgBase64 = img.getString(this.SHEET_IMG_DATA);
  3. if (StringUtils.isBlank(imgBase64))
  4. return;
  5. String[] imgary = imgBase64.split(",");
  6. System.out.println(imgary[0]);
  7. byte[] imgByte = Base64.decodeBase64(imgary[1]);
  8. int imgIdx = wb.addPicture(imgByte, Workbook.PICTURE_TYPE_JPEG);
  9. XSSFDrawing drawImg = ws.createDrawingPatriarch();
  10. XSSFClientAnchor anchor = cHelper.createClientAnchor();
  11. int col = img.getIntValue(this.SHEET_IMG_COL);
  12. int row = img.getIntValue(this.SHEET_IMG_ROW);
  13. anchor.setCol1(col);
  14. anchor.setRow1(row);
  15. XSSFPicture pict = drawImg.createPicture(anchor, imgIdx);
  16. Integer colSpan = img.getInteger(this.SHEET_IMG_COLSPAN);
  17. if (colSpan == null)
  18. colSpan = 1;
  19. Integer rowSpan = img.getInteger(this.SHEET_IMG_ROWSPAN);
  20. if (rowSpan == null)
  21. rowSpan = 1;
  22. pict.resize(colSpan, rowSpan);
  23. }

五、总结

这次通过传入JSON对象生成样式丰富的excel文件,对于POI操作office文档又更加熟悉一些。相对于解析excel文档,生成就不用考虑文件格式,如:兼容2003格式,考虑大文件sax方式解析。相对于js前端生成excel文件,增加了对生成后文件二次加工的可能性,所以在功能入口中,采用了生成二进制流的方式。文件生成好后,可以继续发送邮件,上传ftp等操作。
重点说明

  • 对于各数据区域数据,保持区域数据独立性(数据索引值)
  • 对于图片开始行和开始列,索引值是针对一个完整的sheet
  • 对于表头区域,多表头采用 | 分割,减少部分传输数据
  • excel中style为所有sheet共享样式。

POI生成EXCEL文件的更多相关文章

  1. 【原创】POI 生成Excel文件并下载

    ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 效果图: 实现 1.在pom中添加依赖: <!-- https://mvnrepository.com/artifact/org.apa ...

  2. java利用poi生成excel文件后下载本地

    1.该功能需要poi的jar包,链接: http://pan.baidu.com/s/1migAtNq 密码: 38fx. 2.首先新建一个实体类,用以存放单个数据 public class Test ...

  3. Java 利用 poi 生成 Excel文件的通用例子

    在用java 写数据库应用的时候, 通常会生成各种报表,而这些报表可能会被导出为各种格式的文件,比如Excel文档,pdf 文档等等. 今天先做了一个生成Excel 文档的例子,主要解决以下问题: 生 ...

  4. POI生成EXCEL文件(字体、样式、单元格合并、计算公式)

    创建一个封装类: package com.jason.excel; import java.io.FileNotFoundException; import java.io.FileOutputStr ...

  5. 使用POI生成Excel文件,可以自动调整excel列宽

    //autoSizeColumn()方法自动调整excel列宽 importjava.io.FileOutputStream; importorg.apache.poi.hssf.usermodel. ...

  6. Jquery的一键上传组件OCUpload及POI解析Excel文件

    第一步:将js文件引入页面 <script type="text/javascript" src="${pageContext.request.contextPat ...

  7. Java Struts2 POI创建Excel文件并实现文件下载

    Java Struts2 POI创建Excel文件并实现文件下载2013-09-04 18:53 6059人阅读 评论(1) 收藏 举报 分类: Java EE(49) Struts(6) 版权声明: ...

  8. 使用POI生成Excel报表

    先把报表模板截图贴上来 下面是POI编写的报表生成类ExcelReport.java package com.jadyer.report; import java.io.FileNotFoundExc ...

  9. 实现excel导入导出功能,excel导入数据到页面中,页面数据导出生成excel文件

    今天接到项目中的一个功能,要实现excel的导入,导出功能.这个看起来思路比较清楚,但是做起了就遇到了不少问题. 不过核心的问题,大家也不会遇到了.每个项目前台页面,以及数据填充方式都不一样,不过大多 ...

随机推荐

  1. ansible小技巧

    出现带'u'的 unicode编码, 在python里 .encode()回去,尤其是经过shell处理的,最好在shell中使用jinja2来处理一下 最好这种情况在shell模块使用jinja2的 ...

  2. st9720-GB 中文编码对照表

    做嵌入式LED 屏幕显示的时候,需要ST9720 中文编码,网上找了好几个版本,版本不同居然对应的code也不同!找了一个靠谱的pdf版本编码对照表,供大家参考 ..... ..... 下载地址: h ...

  3. HangFire快速入门 分布式后端作业调度框架服务

    安装 NuGet 上有几个可用的Hangfire 的软件包.如果在ASP.NET应用程序中安装HangFire,并使用Sql Server作为存储器,那么请在Package Manager Conso ...

  4. Node.js 操作 OSX 系统麦克风、扬声器音量

    最近几年 Electron 很火,公司也正好有个项目想做跨平台客户端,大家研究了一下就选择了 Electron,第一次做 js 的项目遇到了不少坑,不过也都一点点解决了. 因为项目中需要对用户录音,H ...

  5. pwn易忘操作原理笔记

    堆溢出漏洞: 一.null-byte-off-by-one 漏洞原理:由于输入操作失误,导致可以把size最低字节修改为\x00,overlapchunk利用. 构造 1.freeB,此时C的pres ...

  6. sklearn库 线性回归库 LinearRegression

    import numpy as np import sklearn.datasets #加载原数据 from sklearn.model_selection import train_test_spl ...

  7. 在WINDOWS中安装使用SIGPACK(MinGW64+Sublime Text3 &Visual Studio)

    本文介绍在Windows下安装使用SigPack库,涉及Armadillo.OpenBLAS安装使用说明,最终实现对SigPack示例CPP基于MinGW64在Sublime Text3下的编译运行, ...

  8. 2018-2019 ICPC, NEERC, Southern Subregional Contest

    目录 2018-2019 ICPC, NEERC, Southern Subregional Contest (Codeforces 1070) A.Find a Number(BFS) C.Clou ...

  9. django默认模板引擎和jinja2模板引擎

    在使用中,大家会发现django默认模板引擎有很多局限性,最明显的就是四则运算.就只能加减,乘除都不支持.另外还有判断相等,不能直接if,要用ifequal.确实不太方便.还有一点,django默认模 ...

  10. HBase shell scan 过滤器用法总结

    比较器: 前面例子中的regexstring:2014-11-08.*.binary:\x00\x00\x00\x05,这都是比较器.HBase的filter有四种比较器: (1)二进制比较器:如’b ...