JFreeChart 生成Word图表

文章目录

  • 1 思路
    • 1.1 概述
    • 1.2 支持的图表类型
    • 1.3 特性
  • 2 准备模板
  • 3 导入依赖
  • 4 图表生成工具类 ChartWithChineseExample
    • 步骤 1: 准备字体文件
    • 步骤 2: 注册字体到`FontFactory`
    • 步骤 3: 设置图表具体位置的字体
      • 柱状图:
      • 饼图:
      • 折线图:
      • 完整代码:
  • 5 业务层 OfficeServicel
  • 6 通用工具类 OfficeUtils
  • 7 控制层 OfficeController
  • 8 导出效果

1 思路

JFreeChart

JFreeChart是一个开源的Java图表库,专为JAVA平台设计,用于生成高质量的2D图表。

1.1 概述

  • JFreeChart是一个完全使用JAVA语言编写的图表绘制类库。
  • 它最初由David Gilbert创建,自2001年以来一直在持续开发和更新,目前已成为Java社区中广泛使用的图表库之一。
  • JFreeChart是一个开源项目,遵循GNU通用公共许可证(LGPL),允许在专有应用程序中使用。

1.2 支持的图表类型

  • JFreeChart支持多种图表类型,包括但不限于:
    • 饼图(Pie charts)
    • 柱状图(Bar charts)
    • 散点图(Scatter plots)
    • 时序图(Time series)
    • 甘特图(Gantt charts)
    • 线形图(Line charts)
    • 气泡图(Bubble charts)
    • 热力图(Heatmaps)

1.3 特性

  • 定制能力:提供大量的定制选项,包括颜色、字体、标签、图例、网格线、数据点等,以满足各种设计需求。
  • 数据源:接受各种数据结构作为输入,如数组、列表或CategoryDataset和TimeSeriesDataset对象。
  • 输出类型:支持多种输出类型,包括Swing组件、图像文件(PNG、JPEG)、矢量图形文件格式(PDF、EPS、SVG)等。
  • 交互性:具有一定的交互功能,如缩放、平移等。

通过 JFreeChart 创建图表,将图表转换为图像格式(如PNG或JPEG),然后将图像解析成InputStream 写入到Word文档的相应位置中。

2 准备模板

在这里插入图片描述

3 导入依赖

        <dependency><groupId>org.jfree</groupId><artifactId>jfreechart</artifactId><version>1.5.3</version></dependency>

4 图表生成工具类 ChartWithChineseExample

在使用org.jfree.chart库生成图表时,如果遇到中文无法正常显示的问题,通常是字体设置的问题。JFreeChart默认使用的字体可能不支持中文字符。要解决这个问题,你需要指定一个支持中文的字体。以下是解决此问题的一般步骤:

步骤 1: 准备字体文件

首先,你需要一个支持中文的TrueType字体文件(.ttf),如宋体(SimSun.ttf)、微软雅黑(msyh.ttf)等。这些字体文件通常可以在Windows系统的C:\Windows\Fonts目录下找到,或者你可以从互联网上下载。

字体文件包可以从这里下载:office字体文件包

步骤 2: 注册字体到FontFactory

在你的Java程序中,使用FontFactory.register()方法注册你的中文字体文件。例如,如果你有SimSun.ttf这个字体文件,可以这样做:

    /*** 注册中文字体*/public static void registerChineseFont() {// 注册中文字体(这里假设已经将字体文件放置在项目的resources目录下)InputStream fontStream = ChartWithChineseExample.class.getResourceAsStream("/font/SIMSUN.TTC"); // 路径根据实际情况调整try {Font customFont = Font.createFont(Font.TRUETYPE_FONT, fontStream).deriveFont(Font.PLAIN, 12);GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();ge.registerFont(customFont);} catch (FontFormatException | IOException e) {e.printStackTrace();}}

步骤 3: 设置图表具体位置的字体

柱状图:

        // 示例字体为宋体,常规,14号Font axisLabelFont = new Font("SimSun", Font.PLAIN, 14);// X轴chart.getCategoryPlot().getDomainAxis().setLabelFont(axisLabelFont);chart.getCategoryPlot().getDomainAxis().setTickLabelFont(axisLabelFont);// Y轴chart.getCategoryPlot().getRangeAxis().setLabelFont(axisLabelFont);chart.getCategoryPlot().getRangeAxis().setTickLabelFont(axisLabelFont);

饼图:

        chart.getTitle().setFont(new Font("SimSun", Font.BOLD, 18));chart.getLegend().setItemFont(new Font("SimSun", Font.PLAIN, 12));// 获取饼图的plot对象,以便进行进一步定制PiePlot3D plot = (PiePlot3D) chart.getPlot();// 设置标签字体plot.setLabelFont(new Font("SimSun", Font.PLAIN, 14));// 设置无数据信息字体(如果需要)plot.setNoDataMessageFont(new Font("SimSun", Font.PLAIN, 18));

折线图:

        // 设置字体chart.getTitle().setFont(new Font("SimSun", Font.BOLD, 18));chart.getLegend().setItemFont(new Font("SimSun", Font.PLAIN, 12));CategoryPlot plot = (CategoryPlot) chart.getPlot();plot.getDomainAxis().setLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getRangeAxis().setLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getDomainAxis().setTickLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getRangeAxis().setTickLabelFont(new Font("SimSun", Font.PLAIN, 12));

完整代码:

package com.example.demo.uitls;import lombok.extern.slf4j.Slf4j;
import org.jfree.chart.ChartFactory;
import org.jfree.chart.JFreeChart;
import org.jfree.chart.plot.CategoryPlot;
import org.jfree.chart.plot.PiePlot3D;
import org.jfree.chart.plot.PlotOrientation;
import org.jfree.data.category.DefaultCategoryDataset;
import org.jfree.data.general.DefaultPieDataset;
import org.springframework.stereotype.Component;import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;/*** ChartWithChineseExample : 图表生成工具类** @author zyw* @create 2024-06-25  16:20*/@Slf4j
@Component
public class ChartWithChineseExample {// 柱状图临时文件名public final static String BAR_CHART_FILE_NAME = "BAR_CHART.png";// 饼图临时文件名public final static String PIE_CHART_FILE_NAME = "PIE_CHART.png";// 折线图临时文件名public final static String LINE_CHART_FILE_NAME = "LINE_CHART.png";public static InputStream lineChartGeneration(String title, String x, String y, DefaultCategoryDataset dataset) {registerChineseFont();JFreeChart chart = ChartFactory.createLineChart(title, // 图表标题x,       // X轴标签y,         // Y轴标签dataset,      // 数据集PlotOrientation.VERTICAL, // 图表方向true,        // 是否显示图例true,        // 是否生成工具提示false        // 是否生成URL链接);chart.getTitle().setFont(new Font("SimSun", Font.BOLD, 18));chart.getLegend().setItemFont(new Font("SimSun", Font.PLAIN, 14));// 示例字体为宋体,常规,14号Font axisLabelFont = new Font("SimSun", Font.PLAIN, 14);// X轴chart.getCategoryPlot().getDomainAxis().setLabelFont(axisLabelFont);chart.getCategoryPlot().getDomainAxis().setTickLabelFont(axisLabelFont);// Y轴chart.getCategoryPlot().getRangeAxis().setLabelFont(axisLabelFont);chart.getCategoryPlot().getRangeAxis().setTickLabelFont(axisLabelFont);try {// 将图表转换为字节数组ByteArrayOutputStream outputStream = new ByteArrayOutputStream();BufferedImage chartImage = chart.createBufferedImage(400, 300); // 设置图像的宽高ImageIO.write(chartImage, "png", outputStream);byte[] chartBytes = outputStream.toByteArray();// 将字节数组转换为InputStreamInputStream inputStream = new ByteArrayInputStream(chartBytes);return inputStream;} catch (IOException e) {log.error("折线图图生成异常");return null;}}/*** 饼图生成** @param title   标题* @param dataset 数据集* @return*/public static InputStream pieChartGeneration(String title, DefaultPieDataset dataset) {registerChineseFont();// 使用数据集创建饼图JFreeChart chart = ChartFactory.createPieChart3D(title, // 图表标题dataset, // 数据集true, // 是否显示图例true, // 是否生成工具提示false // 是否生成URL链接);chart.getTitle().setFont(new Font("SimSun", Font.BOLD, 18));chart.getLegend().setItemFont(new Font("SimSun", Font.PLAIN, 12));// 获取饼图的plot对象,以便进行进一步定制PiePlot3D plot = (PiePlot3D) chart.getPlot();// 设置标签字体plot.setLabelFont(new Font("SimSun", Font.PLAIN, 14));// 设置无数据信息字体(如果需要)plot.setNoDataMessageFont(new Font("SimSun", Font.PLAIN, 18));try {// 将图表转换为字节数组ByteArrayOutputStream outputStream = new ByteArrayOutputStream();BufferedImage chartImage = chart.createBufferedImage(400, 300); // 设置图像的宽高ImageIO.write(chartImage, "png", outputStream);byte[] chartBytes = outputStream.toByteArray();// 将字节数组转换为InputStreamInputStream inputStream = new ByteArrayInputStream(chartBytes);return inputStream;} catch (IOException e) {log.error("饼图生成异常");return null;}}/*** 注册中文字体*/public static void registerChineseFont() {// 注册中文字体(这里假设已经将字体文件放置在项目的resources目录下)InputStream fontStream = ChartWithChineseExample.class.getResourceAsStream("/font/SIMSUN.TTC"); // 路径根据实际情况调整try {Font customFont = Font.createFont(Font.TRUETYPE_FONT, fontStream).deriveFont(Font.PLAIN, 12);GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();ge.registerFont(customFont);} catch (FontFormatException | IOException e) {e.printStackTrace();}}/*** 创建柱状图表** @param dataset 数据集* @return*/public static InputStream createChartPanel(String title, String x, String y, DefaultCategoryDataset dataset) {registerChineseFont();// 创建图表JFreeChart chart = ChartFactory.createBarChart(title, // 图表标题x, // X轴标签y, // Y轴标签dataset,PlotOrientation.VERTICAL,true, // 是否显示图例true, // 是否使用工具提示false // 是否生成URL链接);// 设置字体chart.getTitle().setFont(new Font("SimSun", Font.BOLD, 18));chart.getLegend().setItemFont(new Font("SimSun", Font.PLAIN, 12));CategoryPlot plot = (CategoryPlot) chart.getPlot();plot.getDomainAxis().setLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getRangeAxis().setLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getDomainAxis().setTickLabelFont(new Font("SimSun", Font.PLAIN, 12));plot.getRangeAxis().setTickLabelFont(new Font("SimSun", Font.PLAIN, 12));try {// 将图表转换为字节数组ByteArrayOutputStream outputStream = new ByteArrayOutputStream();BufferedImage chartImage = chart.createBufferedImage(400, 300); // 设置图像的宽高ImageIO.write(chartImage, "png", outputStream);byte[] chartBytes = outputStream.toByteArray();// 将字节数组转换为InputStreamInputStream inputStream = new ByteArrayInputStream(chartBytes);return inputStream;} catch (IOException e) {log.error("柱状图生成异常");return null;}}}

5 业务层 OfficeServicel

在word中遍历所有段落,找到需要插入图表的段落索引。

此处省略上诉已展示代码。

/*** OfficeServiceImpl :** @author zyw* @create 2024-06-24  15:41*/
@Service
@Slf4j
public class OfficeServiceImpl implements OfficeService {private static final String HEADER_2_1 = "营养成分摄入比例";private static final String HEADER_2_2 = "心率血氧检查";private static final String HEADER_2_3 = "睡眠质量趋势";@Overridepublic XWPFDocument getHealthReport(HealthReportQuery query) {try {FileInputStream fileInputStream = SpringUtils.convertInputStreamToFileInputStream(resourceLoader.getResource(PERSONAL_HEALTH_REPORT_TEMPLATE).getInputStream());XWPFDocument xwpfDocument = new XWPFDocument(fileInputStream);// 插入历史体重int index5 = OfficeUtils.findParagraphIndexByText(xwpfDocument, HEADER_2_1);insertChartOne(xwpfDocument, index5);// 插入心率检查int index6 = OfficeUtils.findParagraphIndexByText(xwpfDocument, HEADER_2_2);insertChartTwo(xwpfDocument, index6);// 插入睡眠质量趋势int index7 = OfficeUtils.findParagraphIndexByText(xwpfDocument, HEADER_2_3);insertChartThree(xwpfDocument, index7);return xwpfDocument;} catch (Exception e) {log.info("获取健康报告失败", e);return null;}}/*** 获取文本在文档中的索引** @param doc  文档* @param text 文本标识* @return*/public static int findParagraphIndexByText(XWPFDocument doc, String text) {// 获取所有段落List<XWPFParagraph> paragraphs = doc.getParagraphs();// 查找目标段落int targetParagraphIndex = -1;for (int i = 0; i < paragraphs.size(); i++) {if (paragraphs.get(i).getText().contains(text)) {targetParagraphIndex = i;break;}}return targetParagraphIndex;}/*** 插入图表 1** @param document* @param index* @throws Exception*/public void insertChartOne(XWPFDocument document, Integer index) throws Exception {// 填充图表数据DefaultPieDataset<String> dataset = new DefaultPieDataset<String>();dataset.setValue("碳水化合物(30%)", 30);dataset.setValue("蛋白质(30%)", 30);dataset.setValue("脂肪(25%)", 25);dataset.setValue("纤维等营养素(15%)", 15);// 创建图表示例InputStream chartPanel = ChartWithChineseExample.pieChartGeneration("营养成分摄入比例", dataset);// 获取所有段落List<XWPFParagraph> paragraphs = document.getParagraphs();// 在目标段落后添加一个新的段落XWPFParagraph paragraph = document.insertNewParagraph(paragraphs.get(index + 1).getCTP().newCursor().newCursor());// 设置段落的样式和属性,实现换行paragraph.setWordWrap(true); // 设置自动换行// 设置段落水平居中paragraph.setAlignment(ParagraphAlignment.CENTER);// 设置段落内文字(这里是空格)垂直居中paragraph.setVerticalAlignment(TextAlignment.CENTER);// 调整行距以确保图片上下居中,这一步可能需要根据实际情况调整XWPFRun run = paragraph.createRun();run.addPicture(chartPanel, XWPFDocument.PICTURE_TYPE_PNG, ChartWithChineseExample.BAR_CHART_FILE_NAME, Units.toEMU(400), Units.toEMU(300));}/*** 插入图表2 心率血氧** @param document* @param index*/public void insertChartTwo(XWPFDocument document, Integer index) throws Exception {// 填充图表数据DefaultCategoryDataset dataset = new DefaultCategoryDataset();dataset.addValue(77, "心率", "2024-06-23");dataset.addValue(85, "心率", "2024-06-24");dataset.addValue(99, "心率", "2024-06-25");dataset.addValue(92.76, "血氧饱和度", "2024-06-23");dataset.addValue(98.74, "血氧饱和度", "2024-06-24");dataset.addValue(94.2, "血氧饱和度", "2024-06-25");// 创建图表示例InputStream chartPanel = ChartWithChineseExample.createChartPanel("心率和血氧饱和度图表", "日期", "心率(次/分)、血氧饱和度(%)", dataset);// 获取所有段落List<XWPFParagraph> paragraphs = document.getParagraphs();// 在目标段落后添加一个新的段落XWPFParagraph paragraph = document.insertNewParagraph(paragraphs.get(index + 1).getCTP().newCursor().newCursor());// 设置段落的样式和属性,实现换行paragraph.setWordWrap(true); // 设置自动换行// 设置段落水平居中paragraph.setAlignment(ParagraphAlignment.CENTER);// 设置段落内文字(这里是空格)垂直居中paragraph.setVerticalAlignment(TextAlignment.CENTER);// 调整行距以确保图片上下居中,这一步可能需要根据实际情况调整XWPFRun run = paragraph.createRun();run.addPicture(chartPanel, XWPFDocument.PICTURE_TYPE_PNG, ChartWithChineseExample.PIE_CHART_FILE_NAME, Units.toEMU(400), Units.toEMU(300));}/*** 插入图表3 睡眠质量趋势** @param document* @param index* @throws Exception*/public void insertChartThree(XWPFDocument document, Integer index) throws Exception {// 填充图表数据DefaultCategoryDataset dataset = new DefaultCategoryDataset();dataset.addValue(7.8, "起床时间", "06/18");dataset.addValue(8, "起床时间", "06/19");dataset.addValue(7.5, "起床时间", "06/20");dataset.addValue(8.3, "起床时间", "06/21");dataset.addValue(9, "起床时间", "06/22");dataset.addValue(9.5, "起床时间", "06/23");dataset.addValue(23, "睡眠时间", "06/18");dataset.addValue(24, "睡眠时间", "06/19");dataset.addValue(22.6, "睡眠时间", "06/20");dataset.addValue(23.2, "睡眠时间", "06/21");dataset.addValue(21.8, "睡眠时间", "06/22");dataset.addValue(23.7, "睡眠时间", "06/23");// 创建图表示例InputStream chartPanel = ChartWithChineseExample.lineChartGeneration("睡眠质量趋势", "日期", "睡眠时间", dataset);// 获取所有段落List<XWPFParagraph> paragraphs = document.getParagraphs();// 在目标段落后添加一个新的段落XWPFParagraph paragraph = OfficeUtils.insertNewParagraph(paragraphs, document,index);// 设置段落的样式和属性,实现换行paragraph.setWordWrap(true); // 设置自动换行// 设置段落水平居中paragraph.setAlignment(ParagraphAlignment.CENTER);// 设置段落内文字(这里是空格)垂直居中paragraph.setVerticalAlignment(TextAlignment.CENTER);// 调整行距以确保图片上下居中,这一步可能需要根据实际情况调整XWPFRun run = paragraph.createRun();run.addPicture(chartPanel, XWPFDocument.PICTURE_TYPE_PNG, ChartWithChineseExample.LINE_CHART_FILE_NAME, Units.toEMU(400), Units.toEMU(300));}
}

6 通用工具类 OfficeUtils

package com.example.demo.uitls;import jakarta.servlet.http.HttpServletResponse;
import org.apache.poi.xwpf.usermodel.*;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.*;import java.lang.reflect.Field;
import java.io.*;
import java.math.BigInteger;
import java.util.*;/*** OfficeUtils : Office工具类** @author zyw* @create 2024-06-24  16:35*/public class OfficeUtils {/*** 插入新段落** @param paragraphs 段落集合* @param document   文档* @param index      插入位置* @return 新段落*/public static XWPFParagraph insertNewParagraph(List<XWPFParagraph> paragraphs, XWPFDocument document, Integer index) {if (paragraphs.size() == index + 1) {return document.createParagraph();} else {return document.insertNewParagraph(paragraphs.get(index + 1).getCTP().newCursor().newCursor());}}/*** 设置表格宽度去除边框** @param table 表格* @param width 宽度值*/public static void setTableWidthToRemoveBorder(XWPFTable table, Integer width) {// 去除表格边框CTTblPr tblPr2 = table.getCTTbl().getTblPr();CTTblBorders borders2 = tblPr2.addNewTblBorders();borders2.addNewBottom().setVal(STBorder.NONE);borders2.addNewTop().setVal(STBorder.NONE);borders2.addNewLeft().setVal(STBorder.NONE);borders2.addNewRight().setVal(STBorder.NONE);borders2.addNewInsideH().setVal(STBorder.NONE);borders2.addNewInsideV().setVal(STBorder.NONE);// 设置表格整体样式tblPr2.addNewTblW().setW(BigInteger.valueOf(width)); // 设置表格宽度}/*** 设置表格单元格宽度及文本居中** @param cell  单元格* @param width 宽度占比*/public static void setTheLandscapeHeader(XWPFTableCell cell, double width) {setsTheCellWidth(cell, width);// 获取单元格属性对象CTTcPr tcPr = cell.getCTTc().isSetTcPr() ? cell.getCTTc().getTcPr() : cell.getCTTc().addNewTcPr();// 设置垂直对齐方式为居中CTVerticalJc vJc = tcPr.isSetVAlign() ? tcPr.getVAlign() : tcPr.addNewVAlign();vJc.setVal(STVerticalJc.CENTER);}/*** 设置表格单元格宽度样式靠左** @param cell  单元格* @param width 宽度占比*/public static void setsTheCellWidthLeft(XWPFTableCell cell, double width) {// 假设A4纸宽约为210mm,1mm=360EMU,则A4宽约为7920EMUint emuFor30Percent = (int) (7920 * width);CTTblWidth ctTblWidth = cell.getCTTc().addNewTcPr().addNewTcW();// 设置宽度为2000EMU,你可以根据需要调整这个值ctTblWidth.setW(BigInteger.valueOf(emuFor30Percent));// 设置宽度类型为字符单位(也可以是其他单位,如百分比等)ctTblWidth.setType(STTblWidth.PCT);// 设置垂直居中cell.setVerticalAlignment(XWPFTableCell.XWPFVertAlign.CENTER);for (XWPFParagraph para : cell.getParagraphs()) {//居中para.setAlignment(ParagraphAlignment.LEFT);}}/*** 设置表格单元格宽度** @param cell  单元格* @param width 宽度占比*/public static void setsTheCellWidth(XWPFTableCell cell, double width) {// 假设A4纸宽约为210mm,1mm=360EMU,则A4宽约为7920EMUint emuFor30Percent = (int) (7920 * width);CTTblWidth ctTblWidth = cell.getCTTc().addNewTcPr().addNewTcW();// 设置宽度为2000EMU,你可以根据需要调整这个值ctTblWidth.setW(BigInteger.valueOf(emuFor30Percent));// 设置宽度类型为字符单位(也可以是其他单位,如百分比等)ctTblWidth.setType(STTblWidth.PCT);// 设置垂直居中cell.setVerticalAlignment(XWPFTableCell.XWPFVertAlign.CENTER);for (XWPFParagraph para : cell.getParagraphs()) {//居中para.setAlignment(ParagraphAlignment.CENTER);}}/*** 设置表格行的高度** @param row      行* @param heightCm 高度占比*/public static void setRowHeight(XWPFTableRow row, double heightCm) {int emuForHeight = (int) (360 * heightCm);CTTrPr trPr = row.getCtRow().addNewTrPr();CTHeight ht = trPr.addNewTrHeight();ht.setVal(BigInteger.valueOf(emuForHeight));}/*** 创建表格行** @param table 表格* @param index 行索引* @return*/public static XWPFTableRow createRow(XWPFTable table, int index) {return Objects.isNull(table.getRow(index)) ? table.createRow() : table.getRow(index);}/*** 创建单元格** @param row   行* @param index 列索引* @return*/public static XWPFTableCell createCell(XWPFTableRow row, int index) {return Objects.isNull(row.getCell(index)) ? row.createCell() : row.getCell(index);}/*** 获取文本在文档中的索引** @param doc  文档* @param text 文本标识* @return*/public static int findParagraphIndexByText(XWPFDocument doc, String text) {// 获取所有段落List<XWPFParagraph> paragraphs = doc.getParagraphs();// 查找目标段落int targetParagraphIndex = -1;for (int i = 0; i < paragraphs.size(); i++) {if (paragraphs.get(i).getText().contains(text)) {targetParagraphIndex = i;break;}}return targetParagraphIndex;}/*** 对象转Map** @param obj 对象* @return*/public static Map<String, String> objectToMap(Object obj) {Map<String, String> map = new HashMap<>();Class<?> clazz = obj.getClass();// 获取类中所有声明的字段(包括私有、受保护、默认、公共)Field[] fields = clazz.getDeclaredFields();for (Field field : fields) {field.setAccessible(true); // 设置字段可访问(如果是私有的)try {Object value = field.get(obj);String key = "${" + field.getName() + "}"; // 构造key,以${name}形式map.put(key, String.valueOf(value));} catch (IllegalAccessException e) {e.printStackTrace();}}return map;}/*** 段落文本填充** @param document      文档* @param insertTextMap 填充内容*/public static void paragraphTextFilling(XWPFDocument document, Map<String, String> insertTextMap) {Set<String> set = insertTextMap.keySet();Iterator<XWPFParagraph> itPara = document.getParagraphsIterator();while (itPara.hasNext()) {// 获取文档中当前的段落文字信息XWPFParagraph paragraph = itPara.next();List<XWPFRun> run = paragraph.getRuns();// 遍历段落文字对象for (int i = 0; i < run.size(); i++) {// 获取段落对象if (run.get(i) == null) {    //段落为空跳过continue;}String sectionItem = null;try {// 检查段落中是否包含文本框sectionItem = run.get(i).getText(run.get(i).getTextPosition());    //段落内容} catch (Exception e) {}if (sectionItem == null) {continue;}// 遍历自定义表单关键字,替换Word文档中的内容Iterator<String> iterator = set.iterator();while (iterator.hasNext()) {// 当前关键字String key = iterator.next();// 替换内容sectionItem = sectionItem.replace(key, String.valueOf(insertTextMap.get(key)));}run.get(i).setText(sectionItem, 0);}}}/*** 处理Word响应** @param downloadName 下载文件名* @param inputStream  文件输入流* @param response     响应*/public static void processingWordResponses(String downloadName,InputStream inputStream,HttpServletResponse response) {try {// 设置响应的Content-Typeresponse.setContentType("application/octet-stream");response.setCharacterEncoding("utf-8");// 设置Content-Disposition头部,指示浏览器下载文件,文件名为document.docxdownloadName = new String(downloadName.getBytes("UTF-8"), "ISO-8859-1");response.setHeader("Content-Disposition", "attachment;filename=" + downloadName + ".docx");// 获取响应的输出流OutputStream outputStream = response.getOutputStream();byte[] buffer = new byte[4096];int bytesRead = -1;// 将InputStream中的内容写入到OutputStream中while ((bytesRead = inputStream.read(buffer)) != -1) {outputStream.write(buffer, 0, bytesRead);}// 关闭流inputStream.close();outputStream.close();} catch (Exception e) {}}/*** word转InputStream** @param document* @return*/public static InputStream writeDocumentToInputStream(XWPFDocument document) {try {ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();document.write(byteArrayOutputStream);byteArrayOutputStream.close();return new ByteArrayInputStream(byteArrayOutputStream.toByteArray());} catch (IOException e) {e.printStackTrace();return null;}}
}

7 控制层 OfficeController

package com.example.demo.controller;import com.example.demo.dto.HealthReportQuery;
import com.example.demo.service.OfficeService;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.Parameters;
import io.swagger.v3.oas.annotations.enums.ParameterIn;
import io.swagger.v3.oas.annotations.tags.Tag;
import jakarta.annotation.Resource;
import jakarta.servlet.http.HttpServletResponse;
import org.springframework.web.bind.annotation.*;/*** OfficeController : Office办公文件控制器** @author zyw* @create 2024-06-24  15:40*/@Tag(name = "Office办公文件控制器")
@RestController
@RequestMapping("/office")
public class OfficeController {@Resourceprivate OfficeService officeService;@GetMapping("/getHealthReportWord")@Operation(summary = "获取健康报告Word", description = "获取健康报告")@Parameters({@Parameter(name = "name", description = "姓名", required = true, in = ParameterIn.QUERY),@Parameter(name = "gender", description = "性别", required = true, in = ParameterIn.QUERY),@Parameter(name = "age", description = "年龄", required = true, in = ParameterIn.QUERY)})public void getHealthReportWord(HealthReportQuery query, HttpServletResponse response) {officeService.getHealthReportWord(officeService.getHealthReport(query), query, response);}}

在这里插入图片描述

8 导出效果

Word效果:
在这里插入图片描述

PDF效果:
在这里插入图片描述

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

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

相关文章

国产车规MCU OTA方案总结

目录 1. 旗芯微FC4150 OTA 2. 云途YTM32B1MD OTA 3.小结 今天没有废话&#xff0c;啪一下很快&#xff0c;把目前接触到的国内带eFlash的车规MCU硬件OTA方案做一个梳理。 1. 旗芯微FC4150 OTA 旗芯微FC4150是基于ARM Cortex(快去审核下官网介绍&#xff0c;少了个T)-M4F内…

openGauss Developer Day 2024丨MogDB实现数据库技术跨越,Ustore引擎革新存储新境界

openGauss Developer Day 2024 6月21日&#xff0c;openGauss Developer Day 2024在北京昆泰嘉瑞文化中心成功召开。大会聚集学术专家、行业用户、合作伙伴和开发者&#xff0c;共同探讨数据库面向多场景的技术创新&#xff0c;分享基于 openGauss 的行业联合创新成果及实践案例…

探索PHP中的魔术常量

PHP中的魔术常量&#xff08;Magic Constants&#xff09;是一些特殊的预定义常量&#xff0c;它们在不同的上下文中具有不同的值。这些常量可以帮助开发者获取文件路径、行号、函数名等信息&#xff0c;从而方便调试和日志记录。本文将详细介绍PHP中的魔术常量&#xff0c;帮助…

web前端——javaScript

目录 一、javaScript概述 1.javaScript历史 2.JavaScript与html,css关系 二、基本语法 ①放在head中 ②放在 body中 ③写在外部的.js文件中 1.变量 2.数据类型 3.算术运算符 4.逻辑运算符 5.赋值运算 6.逻辑运算符 7.条件运算符 8.控制语句 三、函数 1…

Arduino - 按钮 - 长按短按

Arduino - Button - Long Press Short Press Arduino - 按钮 - 长按短按 Arduino - Button - Long Press Short Press We will learn: 我们将学习&#xff1a; How to detect the button’s short press 如何检测按钮的短按How to detect the button’s long press 如何检测…

重大进展!微信支付收款码全场景接入银联网络

据中国银联6月19日消息&#xff0c;近日&#xff0c;银联网络迎来微信支付收款码场景的全面接入&#xff0c;推动条码支付互联互通取得新进展&#xff0c;为境内外广大消费者提供更多支付选择、更好支付体验。 2024年6月&#xff0c;伴随微信支付经营收款码的开放&#xff0c;微…

Rust: duckdb和polars读csv文件比较

一、文件准备 样本内容&#xff0c;N行9列的csv标准格式&#xff0c;有字符串&#xff0c;有浮点数&#xff0c;有整型。 有两个csv文件&#xff0c;一个大约是2.1万行&#xff1b;一个是64万行。 二、toml文件 [package] name "my_duckdb" version "0.1.0&…

VSCode安装OpenImageDebugger

VSCode安装OpenImageDebugger 1. 官网2. 编译2.1 依赖项2.2 编译 OpenImageDebugger2.3 配置 GDB 和 LLDB 3. 验证安装是否成功 1. 官网 下载路径&#xff1a;OpenImageDebugger 2. 编译 2.1 依赖项 官网上描述&#xff0c; Qt 5.15.1Python 3.10.12 这两个其实配置并不需…

图解HTTP笔记整理(前六章)

图解HTTP 第一章 web使用HTTP &#xff08;HyperText Transfer Protocol&#xff0c;超文本传输协议&#xff09;协议作文规范&#xff0c;完成从客户端到服务器端等一系列运作流程。 协议&#xff1a;计算机与网络设备要相互通信&#xff0c;双方就必须基于相同的方法。比如…

【论文阅读】--Popup-Plots: Warping Temporal Data Visualization

弹出图&#xff1a;扭曲时态数据可视化 摘要1 引言2 相关工作3 弹出图3.1 椭球模型3.1.1 水平轨迹3.1.2 垂直轨迹3.1.3 组合轨迹 3.2 视觉映射与交互 4 实施5 结果6 评估7 讨论8 结论和未来工作致谢参考文献 期刊: IEEE Trans. Vis. Comput. Graph.&#xff08;发表日期: 2019&…

HQChart使用教程30-K线图如何对接第3方数据41-分钟K线叠加股票增量更新

HQChart使用教程30-K线图如何对接第3方数据40-日K叠加股票增量更新 叠加股票叠加分钟K线更新Request 字段说明Data.symbol 协议截图返回json数据结构overlaydata HQChart代码地址交流 叠加股票 示例地址:https://jones2000.github.io/HQChart/webhqchart.demo/samples/kline_i…

可以一键生成热点营销视频的工具,建议收藏

在当今的商业环境中&#xff0c;热点营销已经成为了一种非常重要的营销策略。那么&#xff0c;什么是热点营销呢&#xff1f;又怎么做热点营销视频呢&#xff1f; 最近高考成绩慢慢公布了&#xff0c;领导让结合“高考成绩公布”这个热点&#xff0c;做一个关于企业或产品的营销…

鸿蒙NEXT开发:工具常用命令—install

安装三方库。 命令格式 ohpm install [options] [[<group>/]<pkg>[<version> | tag:<tag>]] ... ohpm install [options] <folder> ohpm install [options] <har file> alias: i 说明 group&#xff1a;三方库的命名空间&#xff0c;可…

sys.stdin对象——实现标准输入

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 语法参考 sys.stdin是一个标准化输入对象&#xff0c;可以连续输入或读入文件所有内容&#xff0c;不结束&#xff0c;不能直接使用。输入完成后&am…

print()函数——打印输出

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 print()函数是Python编程最常见的函数&#xff0c;常用于输出程序结果&#xff0c;默认输出到屏幕&#xff0c;也可以输出到指定文件。 语法参考 pr…

吉他谱制作软件哪个好 吉他弹唱谱制作软件推荐

在市面上存在着多种吉他谱制作软件&#xff0c;如何选择一款适合自己需求的软件成为了许多人面临的挑战。下面来看看吉他谱制作软件哪个好&#xff0c;吉他弹唱谱制作软件推荐的相关内容。 一、吉他谱制作软件哪个好 吉他谱制作软件在现代音乐创作中扮演着重要角色&#xff0c…

调频信号FM的原理与matlab与FPGA实现

平台&#xff1a;matlab r2021b&#xff0c;vivado2023.1 本文知识内容摘自《软件无线电原理和应用》 调频(FM)是载波的瞬时频率随调制信号成线性变化的一种调制方式&#xff0c;音频调频信号的数学表达式可以写为&#xff1a; Fm频率调制&#xff0c;载波的幅度随着调制波形…

open()函数——打开文件并返回文件对象

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 open()函数用于打开文件&#xff0c;返回一个文件读写对象&#xff0c;然后可以对文件进行相应读写操作。 语法参考 open()函数的语法格式如下&…

【K8s】专题六(2):Kubernetes 稳定性之健康检查

以下内容均来自个人笔记并重新梳理&#xff0c;如有错误欢迎指正&#xff01;如果对您有帮助&#xff0c;烦请点赞、关注、转发&#xff01;欢迎扫码关注个人公众号&#xff01; 目录 一、基本介绍 二、工作原理 三、探针类型 1、存活探针&#xff08;LivenessProbe&#x…

docker入门配置

1、创建配置镜像 由于国内docker连接外网速度慢&#xff0c;采用代理 vi /etc/docker/daemon.json添加以下内容 {"registry-mirrors": ["https://9cpn8tt6.mirror.aliyuncs.com","https://dockerproxy.com","https://hub-mirror.c.163.co…