1. 准备需要的jar。
<!--生成word-->
            <dependency>
                <groupId>org.apache.poi</groupId>
                <artifactId>poi-ooxml</artifactId>
                <version>4.1.2</version>
            </dependency>
  1. controller层
@RequestMapping(value = {"/uploadWord/{id}"}, method = {RequestMethod.GET,RequestMethod.POST})
    public void uploadWord(HttpServletResponse response,@PathVariable("id") Long id) throws Exception {

        // 根据id,查询对应类型的报告
        TdsmsOptSubscribe tdsmsOptSubscribe = subscribeService.selectTdsmsOptSubscribeById(id);
        Optional.ofNullable(tdsmsOptSubscribe).orElseThrow(()-> new ServiceException("报告id["+id+"]不存在!"));

        String wordPath = this.getClass().getClassLoader().getResource("word/").getPath();;
        XWPFDocument xwpfDocument = null;
        FileWordPdfEntity fileWordPdfEntity = null;
        SimpleDateFormat format = new SimpleDateFormat("yyyyMMddHHmmssSSS");
        String timestamp = format.format(new Date());
        String fileName = tdsmsOptSubscribe.getSubscribeTitle()+"_"+timestamp;
        // 文档类型
        if (Objects.equals("1",tdsmsOptSubscribe.getSubscribeType())){
            // linux下无法直接打开jar中模板的位置,必须把模板转换成输入流,生成临时文件再打开。
            InputStream stream = this.getClass().getClassLoader().getResourceAsStream("word/surfTheInternet.docx");
            wordPath+="surfTheInternetTmp.docx";
            tmpFile(stream,wordPath);
            fileWordPdfEntity = setFileWordPdfEntity(tdsmsOptSubscribe,wordPath);
            xwpfDocument = iTdsmReportFormsService.downLoadFileWord(fileWordPdfEntity);

        } else if (Objects.equals("3",tdsmsOptSubscribe.getSubscribeType())){
            // linux下无法直接打开jar中模板的位置,必须把模板转换成输入流,生成临时文件再打开。
            InputStream stream = this.getClass().getClassLoader().getResourceAsStream("word/fileSecurity.docx");
            wordPath+="fileSecurityTmp.docx";
            tmpFile(stream,wordPath);
            fileWordPdfEntity = setFileWordPdfEntity(tdsmsOptSubscribe,wordPath);
            xwpfDocument = iTdsmReportFormsService.downLoadSecurityWord(fileWordPdfEntity);
        }

        // 文档格式,如果是pdf,就需要把生成的word转换成pdf
        if (Objects.equals("1",tdsmsOptSubscribe.getFileFormat())){
            response.reset();
            response.setContentType("application/octet-stream");
            response.setCharacterEncoding("utf-8");
            //response.setContentLength((int) xwpfDocument.length());
            response.setHeader("Content-Disposition",
                    "attachment;filename=" + new String(fileName.getBytes("utf-8"), "ISO-8859-1") + ".docx");
            OutputStream out = response.getOutputStream();
            //将word对象内容写入输出流
            xwpfDocument.write(out);
            out.close();
        } else {
         
            // docx转换成pdf


            String pdfPath = this.getClass().getClassLoader().getResource("word/").getPath();
            //二进制OutputStream
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            //文档写入流
            xwpfDocument.write(baos);
            //OutputStream写入InputStream二进制流
            ByteArrayInputStream in = new ByteArrayInputStream(baos.toByteArray());
            AsposeUtil bean = new AsposeUtil();
            bean.word2pdf(pdfPath,in,tdsmsOptSubscribe.getSubscribeTitle()+timestamp);
            response.setContentType(MediaType.APPLICATION_OCTET_STREAM_VALUE);
            FileUtils.setAttachmentResponseHeader(response,fileName+ ".pdf");
            FileUtils.writeBytes(pdfPath + fileName + ".pdf",response.getOutputStream());
            FileUtils.deleteFile(pdfPath + fileName + ".pdf");
        }

        // 更新订阅次数
        Integer readNum = Objects.isNull(tdsmsOptSubscribe.getReadNum()) ? 0 : (tdsmsOptSubscribe.getReadNum() + 1);
        tdsmsOptSubscribe.setReadNum(readNum);
        subscribeMapper.updateById(tdsmsOptSubscribe);
    }

  /**
     * 生成临时文件
     * @param stream 文件输入流
     * @param wordPath  临时文件地址
     */
    public void tmpFile(InputStream stream, String wordPath) {

        File file = new File(wordPath);
        try {
            //将读取到的类容存储到临时文件中,后面就可以用这个临时文件访问了
            org.apache.commons.io.FileUtils.copyInputStreamToFile(stream,file);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
  1. service层
package com.tdsms.busi.service.impl;

import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import com.tdsms.busi.domain.*;
import com.tdsms.busi.mapper.*;
import com.tdsms.busi.service.ITdsmReportFormsService;
import com.tdsms.busi.utils.CommonFileUtil;
import com.tdsms.common.minio.MinioUtil;
import com.tdsms.common.utils.DateUtils;
import org.apache.commons.compress.utils.Lists;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.poi.ooxml.POIXMLDocument;
import org.apache.poi.openxml4j.exceptions.InvalidFormatException;
import org.apache.poi.openxml4j.opc.OPCPackage;
import org.apache.poi.ss.formula.functions.T;
import org.apache.poi.ss.util.CellRangeAddress;
import org.apache.poi.xddf.usermodel.chart.XDDFChartData;
import org.apache.poi.xddf.usermodel.chart.XDDFDataSource;
import org.apache.poi.xddf.usermodel.chart.XDDFDataSourcesFactory;
import org.apache.poi.xddf.usermodel.chart.XDDFNumericalDataSource;
import org.apache.poi.xssf.usermodel.XSSFCell;
import org.apache.poi.xssf.usermodel.XSSFSheet;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import org.apache.poi.xwpf.usermodel.XWPFChart;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.multipart.commons.CommonsMultipartFile;

import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.OutputStream;
import java.util.*;
import java.util.concurrent.atomic.AtomicReference;
import java.util.stream.Collector;
import java.util.stream.Collectors;


@Service
public class TdsmReportFormsServiceImpl implements ITdsmReportFormsService {


    @Autowired
    private MinioUtil minioUtil;

    @Autowired
    private TdsmsAccountMapper tdsmsAccountMapper;

    @Autowired
    private TdsmsLogsSearchMapper tdsmsLogsSearchMapper;

    @Autowired
    private TdsmsLogsHtmlMapper tdsmsLogsHtmlMapper;

    @Autowired
    private TdsmsLogsImMsgMapper tdsmsLogsImMsgMapper;

    @Autowired
    private TdsmsLogsEmailMapper tdsmsLogsEmailMapper;

    @Autowired
    private TdsmsLogsFileSendMapper tdsmsLogsFileSendMapper;

    @Autowired
    private TdsmsLogsFileOptMapper tdsmsLogsFileOptMapper;

    @Autowired
    private TdsmsLogsScreenMapper tdsmsLogsScreenMapper;

    @Autowired
    private TdsmsLogsPrintMapper tdsmsLogsPrintMapper;

    @Autowired
    private TdsmsLogsClipboardMapper tdsmsLogsClipboardMapper;

    @Override
    public XWPFDocument downLoadFileWord(FileWordPdfEntity fileWordPdfEntity) throws IOException {
        Map<String, String> map = new HashMap<String, String>();//把要替换的内容放入map
        map.put("{startTime}", fileWordPdfEntity.getStartTime());
        map.put("{endTime}", fileWordPdfEntity.getEndTime());
        map.put("{nowTime}", DateUtils.getTime());
        // List<TdsmsAccount> tdsmsAccountsOrgId = tdsmsAccountMapper.selectTdsmsAccountBatchListOrgIds(fileWordPdfEntity.getOrgIds());
        List<TdsmsAccount> tdsmsAccountsUserId = tdsmsAccountMapper.selectTdsmsAccountBatchListUserIds(fileWordPdfEntity.getUserIdList());
        //获取数据对象 用户名
        Set<TdsmsAccount> userTotal = Sets.newHashSet();
        for (TdsmsAccount tdsmsAccount : tdsmsAccountsUserId) {
            userTotal.add(tdsmsAccount);
        }
        //统计总用户数量
        map.put("{allUserTotal}", String.valueOf(fileWordPdfEntity.getUserIdList().size() ));
        //统计新增用户数
        List<TdsmsAccount> newInsertUserId = newInsertData(tdsmsAccountsUserId, fileWordPdfEntity.getStartTime(), fileWordPdfEntity.getEndTime());
        map.put("{addUserTotal}", String.valueOf(newInsertUserId.size()));
        // 日志查询值针对数据对象用户
        String nameList = userTotal.stream().map(TdsmsAccount::getAccountName).distinct().collect(Collectors.joining(",", "", ""));

        //搜索引擎日志查询
        TdsmsLogsSearch tdsmsLogsSearch = new TdsmsLogsSearch();
        tdsmsLogsSearch.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsSearch.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsSearch.setNameList(nameList);
        List<TdsmsLogsSearch> tdsmsLogsSearches = tdsmsLogsSearchMapper.selectTdsmsLogsSearchList(tdsmsLogsSearch);
        //网页浏览日志查询
        TdsmsLogsHtml tdsmsLogsHtml = new TdsmsLogsHtml();
        tdsmsLogsHtml.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsHtml.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsHtml.setNameList(nameList);
        List<TdsmsLogsHtml> tdsmsLogsHtmls = tdsmsLogsHtmlMapper.selectTdsmsLogsHtmlList(tdsmsLogsHtml);
        //即时通讯日志查询
        TdsmsLogsImMsg tdsmsLogsImMsg = new TdsmsLogsImMsg();
        tdsmsLogsImMsg.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsImMsg.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsImMsg.setNameList(nameList);
        List<TdsmsLogsImMsg> tdsmsLogsImMsgs = tdsmsLogsImMsgMapper.selectTdsmsLogsImMsgList(tdsmsLogsImMsg);
        //电子邮件日志查询
        TdsmsLogsEmail tdsmsLogsEmail = new TdsmsLogsEmail();
        tdsmsLogsEmail.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsEmail.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsEmail.setNameList(nameList);
        List<TdsmsLogsEmail> tdsmsLogsEmails = tdsmsLogsEmailMapper.selectTdsmsLogsEmailList(tdsmsLogsEmail);
        //统计日志总数
        map.put("{logTotal}", String.valueOf(tdsmsLogsSearches.size() + tdsmsLogsHtmls.size() + tdsmsLogsImMsgs.size() + tdsmsLogsEmails.size()));
        //告警日志总数
        Integer alarmLogTotal = 0;
        //告警用户数
        LinkedHashMap<String, Integer> alarmUserTotal = Maps.newLinkedHashMap();
        // 上网行为日志
        LinkedHashMap<String, Integer> internetBehavior = Maps.newLinkedHashMap();

        for (TdsmsLogsSearch logsSearch : tdsmsLogsSearches) {
            if ("2".equals(logsSearch.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsSearch.getName())){
                    alarmUserTotal.put(logsSearch.getName(),alarmUserTotal.get(logsSearch.getName())+1);
                } else {
                    alarmUserTotal.put(logsSearch.getName(),1);
                }
            }else {
                // 统计用户上网行为
                if (internetBehavior.containsKey(logsSearch.getName())){
                    internetBehavior.put(logsSearch.getName(),internetBehavior.get(logsSearch.getName())+1);
                } else {
                    internetBehavior.put(logsSearch.getName(),1);
                }
            }
        }
        for (TdsmsLogsHtml logsHtml : tdsmsLogsHtmls) {
            if ("2".equals(logsHtml.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsHtml.getName())){
                    alarmUserTotal.put(logsHtml.getName(),alarmUserTotal.get(logsHtml.getName())+1);
                } else {
                    alarmUserTotal.put(logsHtml.getName(),1);
                }
            }else {
                // 统计用户上网行为
                if (internetBehavior.containsKey(logsHtml.getName())){
                    internetBehavior.put(logsHtml.getName(),internetBehavior.get(logsHtml.getName())+1);
                } else {
                    internetBehavior.put(logsHtml.getName(),1);
                }
            }
        }
        for (TdsmsLogsImMsg logsImMsg : tdsmsLogsImMsgs) {
            if ("2".equals(logsImMsg.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsImMsg.getName())){
                    alarmUserTotal.put(logsImMsg.getName(),alarmUserTotal.get(logsImMsg.getName())+1);
                } else {
                    alarmUserTotal.put(logsImMsg.getName(),1);
                }
            } else {
                // 统计用户上网行为
                if (internetBehavior.containsKey(logsImMsg.getName())){
                    internetBehavior.put(logsImMsg.getName(),internetBehavior.get(logsImMsg.getName())+1);
                } else {
                    internetBehavior.put(logsImMsg.getName(),1);
                }
            }
        }
        for (TdsmsLogsEmail logsEmail : tdsmsLogsEmails) {
            if ("2".equals(logsEmail.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsEmail.getName())){
                    alarmUserTotal.put(logsEmail.getName(),alarmUserTotal.get(logsEmail.getName())+1);
                } else {
                    alarmUserTotal.put(logsEmail.getName(),1);
                }
            } else {
                // 统计用户上网行为
                if (internetBehavior.containsKey(logsEmail.getName())){
                    internetBehavior.put(logsEmail.getName(),internetBehavior.get(logsEmail.getName())+1);
                } else {
                    internetBehavior.put(logsEmail.getName(),1);
                }
            }
        }
        map.put("{alarmLogTotal}",String.valueOf(alarmLogTotal));
        map.put("{alarmUserTotal}",String.valueOf(alarmUserTotal.values().stream().reduce(0,Integer::sum)));
        OPCPackage pack = POIXMLDocument.openPackage(fileWordPdfEntity.getFilePath());//打开word模板文件
        XWPFDocument document = new XWPFDocument(pack);//创建word对象
        // 替换图表数据
        List<XWPFChart> charts = document.getCharts();

        // 告警用户  取前10
        alarmUserTotal = top10(alarmUserTotal);

        // 对用户上网行为  取前10
        internetBehavior = top10(internetBehavior);

        // 对用户网页浏览 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsHtml>> logsHtmls = tdsmsLogsHtmls.stream().collect(Collectors.groupingBy(TdsmsLogsHtml::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对用户搜索引擎 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsSearch>> logsSearches = tdsmsLogsSearches.stream().collect(Collectors.groupingBy(TdsmsLogsSearch::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对用户电子邮件 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsEmail>> logsEmails = tdsmsLogsEmails.stream().collect(Collectors.groupingBy(TdsmsLogsEmail::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对用户即时通讯 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsImMsg>> logsImMsgs = tdsmsLogsImMsgs.stream().collect(Collectors.groupingBy(TdsmsLogsImMsg::getName,TreeMap::new, Collectors.toList())).descendingMap();

        chartsReplace(charts,alarmUserTotal,internetBehavior,top10(logsSearches),top10(logsHtmls),top10(logsImMsgs),top10(logsEmails));
        CommonFileUtil.replaceStrs(document, map);//替换内容

        //将XWPFDocument文件变为MultipartFile
       // MultipartFile file = xwpfDocumentToCommonsMultipartFile(document, fileWordPdfEntity.getFileName()+".docx");
        //上传文件到minio
       // minioUtil.upload(file);
        return document;
    }

    @Override
    public XWPFDocument downLoadSecurityWord(FileWordPdfEntity fileWordPdfEntity) throws IOException {
        Map<String, String> map = new HashMap<String, String>();//把要替换的内容放入map
        map.put("{startTime}", fileWordPdfEntity.getStartTime());
        map.put("{endTime}", fileWordPdfEntity.getEndTime());
        map.put("{nowTime}", DateUtils.getTime());
        List<TdsmsAccount> tdsmsAccountsUserId = tdsmsAccountMapper.selectTdsmsAccountBatchList(fileWordPdfEntity.getUserIdList());
        //获取数据对象 用户名
        Set<TdsmsAccount> userTotal = Sets.newHashSet();
        for (TdsmsAccount tdsmsAccount : tdsmsAccountsUserId) {
            userTotal.add(tdsmsAccount);
        }
        //统计总用户数量
        map.put("{allUserTotal}", String.valueOf(fileWordPdfEntity.getUserIdList().size() ));
        //统计新增用户数
        List<TdsmsAccount> newInsertUserId = newInsertData(tdsmsAccountsUserId, fileWordPdfEntity.getStartTime(), fileWordPdfEntity.getEndTime());
        map.put("{addUserTotal}", String.valueOf(newInsertUserId.size()));
        // 日志查询值针对数据对象用户
        String nameList = userTotal.stream().map(TdsmsAccount::getAccountName).distinct().collect(Collectors.joining(",", "", ""));

        // 文件外发
        TdsmsLogsFileSend tdsmsLogsFileSend = new TdsmsLogsFileSend();
        tdsmsLogsFileSend.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsFileSend.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsFileSend.setNameList(nameList);
        List<TdsmsLogsFileSend> tdsmsLogsFileSends = tdsmsLogsFileSendMapper.selectTdsmsLogsFileSendList(tdsmsLogsFileSend);

        // 文档操作
        TdsmsLogsFileOpt tdsmsLogsFileOpt = new TdsmsLogsFileOpt();
        tdsmsLogsFileOpt.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsFileOpt.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsFileOpt.setNameList(nameList);
        List<TdsmsLogsFileOpt> tdsmsLogsFileOpts = tdsmsLogsFileOptMapper.selectTdsmsLogsFileOptList(tdsmsLogsFileOpt);

        // 屏幕审计
        TdsmsLogsScreen tdsmsLogsScreen = new TdsmsLogsScreen();
        tdsmsLogsScreen.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsScreen.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsScreen.setNameList(nameList);
        List<TdsmsLogsScreen> tdsmsLogsScreens = tdsmsLogsScreenMapper.selectTdsmsLogsScreenList(tdsmsLogsScreen);

        // 打印审计
        TdsmsLogsPrint tdsmsLogsPrint = new TdsmsLogsPrint();
        tdsmsLogsPrint.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsPrint.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsPrint.setNameList(nameList);
        List<TdsmsLogsPrint> tdsmsLogsPrints = tdsmsLogsPrintMapper.selectTdsmsLogsPrintList(tdsmsLogsPrint);

        // 剪贴板审计
        TdsmsLogsClipboard tdsmsLogsClipboard = new TdsmsLogsClipboard();
        tdsmsLogsClipboard.setStartDate(fileWordPdfEntity.getStartTime());
        tdsmsLogsClipboard.setEndDate(fileWordPdfEntity.getEndTime());
        tdsmsLogsClipboard.setNameList(nameList);
        List<TdsmsLogsClipboard> tdsmsLogsClipboards = tdsmsLogsClipboardMapper.selectTdsmsLogsClipboardList(tdsmsLogsClipboard);


        // 统计日志总数
        map.put("{logTotal}", String.valueOf(tdsmsLogsFileSends.size() + tdsmsLogsFileOpts.size() + tdsmsLogsScreens.size() + tdsmsLogsPrints.size()) + tdsmsLogsClipboards.size());
        // 告警日志总数
        Integer alarmLogTotal = 0;
        // 告警用户数
        LinkedHashMap<String, Integer> alarmUserTotal = Maps.newLinkedHashMap();
        // 用户文档审计日志统计
        LinkedHashMap<String, Integer> auditStatistics= Maps.newLinkedHashMap();

        for (TdsmsLogsFileSend logsFileSends : tdsmsLogsFileSends) {
            if ("2".equals(logsFileSends.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsFileSends.getName())){
                    alarmUserTotal.put(logsFileSends.getName(),alarmUserTotal.get(logsFileSends.getName())+1);
                } else {
                    alarmUserTotal.put(logsFileSends.getName(),1);
                }
            }else {
                // 用户文档审计日志统计
                if (auditStatistics.containsKey(logsFileSends.getName())){
                    auditStatistics.put(logsFileSends.getName(),auditStatistics.get(logsFileSends.getName())+1);
                } else {
                    auditStatistics.put(logsFileSends.getName(),1);
                }
            }
        }

        for (TdsmsLogsFileOpt logsFileOpts : tdsmsLogsFileOpts) {
            if ("2".equals(logsFileOpts.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsFileOpts.getName())){
                    alarmUserTotal.put(logsFileOpts.getName(),alarmUserTotal.get(logsFileOpts.getName())+1);
                } else {
                    alarmUserTotal.put(logsFileOpts.getName(),1);
                }
            }else {
                // 用户文档审计日志统计
                if (auditStatistics.containsKey(logsFileOpts.getName())){
                    auditStatistics.put(logsFileOpts.getName(),auditStatistics.get(logsFileOpts.getName())+1);
                } else {
                    auditStatistics.put(logsFileOpts.getName(),1);
                }
            }
        }

        for (TdsmsLogsScreen logsLogsScreens : tdsmsLogsScreens) {
            if ("2".equals(logsLogsScreens.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsLogsScreens.getName())){
                    alarmUserTotal.put(logsLogsScreens.getName(),alarmUserTotal.get(logsLogsScreens.getName())+1);
                } else {
                    alarmUserTotal.put(logsLogsScreens.getName(),1);
                }
            } else {
                // 用户文档审计日志统计
                if (auditStatistics.containsKey(logsLogsScreens.getName())){
                    auditStatistics.put(logsLogsScreens.getName(),auditStatistics.get(logsLogsScreens.getName())+1);
                } else {
                    auditStatistics.put(logsLogsScreens.getName(),1);
                }
            }
        }

        for (TdsmsLogsPrint logsLogsPrints : tdsmsLogsPrints) {
            if ("2".equals(logsLogsPrints.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsLogsPrints.getName())){
                    alarmUserTotal.put(logsLogsPrints.getName(),alarmUserTotal.get(logsLogsPrints.getName())+1);
                } else {
                    alarmUserTotal.put(logsLogsPrints.getName(),1);
                }
            } else {
                // 用户文档审计日志统计
                if (auditStatistics.containsKey(logsLogsPrints.getName())){
                    auditStatistics.put(logsLogsPrints.getName(),auditStatistics.get(logsLogsPrints.getName())+1);
                } else {
                    auditStatistics.put(logsLogsPrints.getName(),1);
                }
            }
        }

        for (TdsmsLogsClipboard logsLogsClipboards : tdsmsLogsClipboards) {
            if ("2".equals(logsLogsClipboards.getLogType())) {
                // 统计告警日志总数
                alarmLogTotal++;
                // 统计告警用户数
                if (alarmUserTotal.containsKey(logsLogsClipboards.getName())){
                    alarmUserTotal.put(logsLogsClipboards.getName(),alarmUserTotal.get(logsLogsClipboards.getName())+1);
                } else {
                    alarmUserTotal.put(logsLogsClipboards.getName(),1);
                }
            } else {
                // 用户文档审计日志统计
                if (auditStatistics.containsKey(logsLogsClipboards.getName())){
                    auditStatistics.put(logsLogsClipboards.getName(),auditStatistics.get(logsLogsClipboards.getName())+1);
                } else {
                    auditStatistics.put(logsLogsClipboards.getName(),1);
                }
            }
        }

        map.put("{alarmLogTotal}",String.valueOf(alarmLogTotal));
        map.put("{alarmUserTotal}",String.valueOf(alarmUserTotal.values().stream().reduce(0,Integer::sum)));
        OPCPackage pack = POIXMLDocument.openPackage(fileWordPdfEntity.getFilePath());//打开word模板文件

       // POIXMLDocument.openPackage()
        XWPFDocument document = new XWPFDocument(pack);//创建word对象
        // 替换图表数据
        List<XWPFChart> charts = document.getCharts();

        // 告警用户  取前10
        alarmUserTotal = top10(alarmUserTotal);

        // 用户文档审计日志统计  取前10
        auditStatistics = top10(auditStatistics);

        // 对文件外发 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsFileSend>> logsFileSend = tdsmsLogsFileSends.stream().collect(Collectors.groupingBy(TdsmsLogsFileSend::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对文档操作 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsFileOpt>> logsFileOpt = tdsmsLogsFileOpts.stream().collect(Collectors.groupingBy(TdsmsLogsFileOpt::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对屏幕审计 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsScreen>> logsScreen = tdsmsLogsScreens.stream().collect(Collectors.groupingBy(TdsmsLogsScreen::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对打印审计 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsPrint>> logsPrint = tdsmsLogsPrints.stream().collect(Collectors.groupingBy(TdsmsLogsPrint::getName,TreeMap::new, Collectors.toList())).descendingMap();

        // 对剪贴板审计 group by 分组 然后排序 取前10
        NavigableMap<String, List<TdsmsLogsClipboard>> logsClipboard = tdsmsLogsClipboards.stream().collect(Collectors.groupingBy(TdsmsLogsClipboard::getName,TreeMap::new, Collectors.toList())).descendingMap();

        chartsReplace(charts, alarmUserTotal, auditStatistics, top10(logsFileSend), top10(logsFileOpt), top10(logsScreen), top10(logsPrint), top10(logsClipboard));
        CommonFileUtil.replaceStrs(document, map);//替换内容

        //将XWPFDocument文件变为MultipartFile
        // MultipartFile file = xwpfDocumentToCommonsMultipartFile(document, fileWordPdfEntity.getFileName()+".docx");
        //上传文件到minio
        // minioUtil.upload(file);
        return document;
    }

    /**
     *
     * @param charts
     * @param alarmUser 告警用户统计
     * @param auditStatistics 用户文档审计日志统计
     * @param logsFileSend 用户文件外发排名
     * @param logsFileOpt 用户文档操作日志
     * @param logsScreen 用户屏幕审计日志
     * @param logsPrint 用户打印审计日志
     * @param logsClipboard 用户截切板审计日志
     */
    private void chartsReplace(List<XWPFChart> charts, LinkedHashMap<String, Integer> alarmUser, LinkedHashMap<String, Integer> auditStatistics,
                               Map<String, Integer> logsFileSend, Map<String, Integer> logsFileOpt, Map<String, Integer> logsScreen,
                               Map<String, Integer> logsPrint, Map<String, Integer> logsClipboard) {
        List<XWPFChart> sortListForXWPFChart = getSortListForXWPFChart(charts);
        xWPFChartReplace(sortListForXWPFChart.get(0),alarmUser);
        xWPFChartReplace(sortListForXWPFChart.get(1),auditStatistics);
        xWPFChartReplace(sortListForXWPFChart.get(2),logsFileSend);
        xWPFChartReplace(sortListForXWPFChart.get(3),logsFileOpt);
        xWPFChartReplace(sortListForXWPFChart.get(4),logsScreen);
        xWPFChartReplace(sortListForXWPFChart.get(5),logsPrint);
        xWPFChartReplace(sortListForXWPFChart.get(6),logsClipboard);

    }

    private LinkedHashMap<String, Integer> top10(LinkedHashMap<String, Integer> alarmUserTotal) {
        LinkedHashMap<String,Integer> alarmUsers = Maps.newLinkedHashMap();
        // 转换成list
        List<Map.Entry<String, Integer>> list = new ArrayList<>(alarmUserTotal.entrySet());
        Collections.sort(list, new Comparator<Map.Entry<String, Integer>>() {
            // 降序排序
            @Override
            public int compare(Map.Entry<String, Integer> o1, Map.Entry<String, Integer> o2) {
                return o2.getValue().compareTo(o1.getValue());
            }
        });
        List<Map.Entry<String, Integer>> collect1 = list.stream().limit(10).collect(Collectors.toList());
        for (Map.Entry<String, Integer> entry : collect1) {
            alarmUsers.put(entry.getKey(),entry.getValue());
        }
        return alarmUsers;
    }

    public<T>  Map<String ,Integer> top10(NavigableMap<String, List<T>> collect) {
        Map<String ,Integer> countGroupBy = new HashMap<>();
        LinkedHashMap<String,Integer> result = Maps.newLinkedHashMap();
        collect.forEach((k,v) -> {
            countGroupBy.put(k,v.size());
        });
        // 转换成list
        List<Map.Entry<String, Integer>> list = new ArrayList<>(countGroupBy.entrySet());
        Collections.sort(list, new Comparator<Map.Entry<String, Integer>>() {
            // 降序排序
            @Override
            public int compare(Map.Entry<String, Integer> o1, Map.Entry<String, Integer> o2) {
                return o2.getValue().compareTo(o1.getValue());
            }
        });
        List<Map.Entry<String, Integer>> collect1 = list.stream().limit(10).collect(Collectors.toList());
        for (Map.Entry<String, Integer> entry : collect1) {
            System.out.println("Key=" + entry.getKey() + ", Value=" + entry.getValue());
            result.put(entry.getKey(),entry.getValue());
        }
        return result;
    }

    public List<TdsmsAccount> newInsertData (List<TdsmsAccount> tdsmsAccounts,String startTime,String endTime) {
        List<TdsmsAccount> tdsmsAccountsOrgIdAdd = Lists.newArrayList();
        for (TdsmsAccount tdsmsAccount : tdsmsAccounts) {
            Date startTimeDate = DateUtils.dateTime(DateUtils.YYYY_MM_DD, startTime);
            Date endTimeDate = DateUtils.dateTime(DateUtils.YYYY_MM_DD, endTime);
            if(startTimeDate.before(tdsmsAccount.getCreateTime()) &&  endTimeDate.after(tdsmsAccount.getCreateTime())) {
                tdsmsAccountsOrgIdAdd.add(tdsmsAccount);
            }
        }
        return tdsmsAccountsOrgIdAdd;
    }

    /**
     *
     * @param charts 查询上网行为图表前10名数据
     * @param alarmUser 告警用户统计
     * @param internetBehavior 用户上网行为次数
     * @param tdsmsLogsSearches 用户搜索引擎排名
     * @param tdsmsLogsHtmls 网页浏览排名
     * @param tdsmsLogsImMsgs 即时通讯排名
     * @param tdsmsLogsEmails 用户电子邮件排名
     */

    public void chartsReplace(List<XWPFChart> charts, Map<String, Integer> alarmUser, Map<String, Integer> internetBehavior, Map<String, Integer> tdsmsLogsSearches
            , Map<String, Integer> tdsmsLogsHtmls, Map<String, Integer> tdsmsLogsImMsgs, Map<String, Integer> tdsmsLogsEmails) {

        List<XWPFChart> sortListForXWPFChart = getSortListForXWPFChart(charts);
        xWPFChartReplace(sortListForXWPFChart.get(0),alarmUser);
        xWPFChartReplace(sortListForXWPFChart.get(1),internetBehavior);
        xWPFChartReplace(sortListForXWPFChart.get(2),tdsmsLogsHtmls);
        xWPFChartReplace(sortListForXWPFChart.get(3),tdsmsLogsSearches);
        xWPFChartReplace(sortListForXWPFChart.get(4),tdsmsLogsEmails);
        xWPFChartReplace(sortListForXWPFChart.get(5),tdsmsLogsImMsgs);

    }

    public void xWPFChartReplace (XWPFChart chart, Map<String, Integer> data) {
        try{
            //排序
            //获取第一个图表
            XSSFWorkbook workbook1 = chart.getWorkbook();
            XSSFSheet sheet1 = workbook1.getSheetAt(0);
            AtomicReference<Integer> count = new AtomicReference<>(0);
            if (data.size() > 0) {
                data.forEach((k, v) -> {
                    count.getAndSet(count.get() + 1);
                    if(count.get() <= data.size()){
                        xSSFSheetReplace(sheet1, count.get(), k, v);
                    }
                });
            }
//            //因为原模板中 只插入了一行数据,所以这里需要用createCell方法新建一个单元格,直接getCell会报空指针
//            XSSFCell cell20 = sheet1.getRow(2).getCell(0);
//            cell20.setCellValue("李四");
//            XSSFCell cell21 = sheet1.getRow(2).getCell(1);
//            cell21.setCellValue(38);

            XDDFChartData chartData1 = chart.getChartSeries().get(0);
            XDDFChartData.Series s1 = chartData1.getSeries().get(0);
            XDDFDataSource cat1 = XDDFDataSourcesFactory.fromStringCellRange(sheet1,
                    new CellRangeAddress(1, 10
                            , 0, 0));
            XDDFNumericalDataSource val1 = XDDFDataSourcesFactory.fromNumericCellRange(sheet1,
                    new CellRangeAddress(1, 10, 1, 1));
            s1.replaceData(cat1, val1);

//        chart1.setTitleText("替换图表测试");
            chart.plot(chartData1);
        }catch (IOException e) {
            e.printStackTrace();
        } catch (InvalidFormatException e) {
            e.printStackTrace();
        }
    }

    public void xSSFSheetReplace(XSSFSheet sheet, Integer count, String name, Integer value) {
        XSSFCell cell01 = sheet.getRow(count).createCell(0);
        cell01.setCellValue(name);
        XSSFCell cell02 = sheet.getRow(count).createCell(1);
        cell02.setCellValue(value);
    }

    /**
     * XWPFDocument 转 MultipartFile(CommonsMultipartFile)
     *
     * @param document 文档对象
     * @param fileName 文件名
     * @return
     */
    public static MultipartFile xwpfDocumentToCommonsMultipartFile(XWPFDocument document, String fileName) {
        //XWPFDocument转FileItem
        FileItemFactory factory = new DiskFileItemFactory(16, null);
        FileItem fileItem = factory.createItem("textField", "text/plain", true, fileName);
        try {
            OutputStream os = fileItem.getOutputStream();
            document.write(os);
            os.close();
            //FileItem转MultipartFile
            MultipartFile multipartFile = new CommonsMultipartFile(fileItem);
            return multipartFile;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

    /**
     * 对图表List重新排序
     * @param affCharts  doc.getCharts()
     * @return
     */
    public List<XWPFChart> getSortListForXWPFChart(List<XWPFChart> affCharts) {
        List<XWPFChart> charts = new ArrayList<>();
        int itNumber = 0; //计数器
        int oldNumber = 0; //入参计数器
        while (itNumber < affCharts.size()){
            // 从oldCharts.get(0)开始检索,获取排序用图表名
            String name = affCharts.get(oldNumber).getPackagePart().getPartName().toString();
            // 获取此图表排序
            String chartsNum = "";// 图表序号
            boolean flag = false; // 上一个是否为数字
            for (int i = 0; i < name.length(); i++) {
                if (chartsNum.equals("") && name.charAt(i)>=48 && name.charAt(i)<=57 ){
                    chartsNum += name.charAt(i);
                    flag = true;
                }else if ( flag && name.charAt(i)>=48 && name.charAt(i)<=57){
                    chartsNum += name.charAt(i);
                    flag = true;
                }else {
                    flag = false;
                }
            }
            //对比图表序号数字
            int thisChartNum = Integer.parseInt(chartsNum);
            if (thisChartNum == itNumber+1){ //如果相等则加入返回list,且itNumber++
                charts.add(affCharts.get(oldNumber));
                itNumber++;
            }

            //入参计数器+1 如果达到最大值则重置为0
            if (oldNumber == affCharts.size()-1){
                oldNumber = 0;
            }else{
                oldNumber++;
            }
        }
        return charts;
    }
}
  1. CommonFileUtil帮助类
package com.tdsms.busi.utils;

import org.apache.poi.xwpf.usermodel.*;
import org.openxmlformats.schemas.wordprocessingml.x2006.main.*;

import java.math.BigInteger;
import java.util.*;

/**
 * 公共文件类
 * */
public class CommonFileUtil {
    /**
     * 替换文档中的内容
     * @param document 文档对象
     * @param map 数据集合
     * */
    public static void replaceStrs(XWPFDocument document, Map<String,String> map){
        replaceStrs(document, map,null);
    }
    /**
     * 替换文档中的内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * */
    public static void replaceStrs(XWPFDocument document, Map<String,String> map, List<String> list){
        /**
         * 对段落中的标记进行替换
         */
        List<XWPFParagraph> parasList = document.getParagraphs();
        replaceInAllParagraphs(parasList, map);
        replaceTableStrs(document, map,list);
    }
    /**
     * 替换段落中的内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * @param replaceList 使用指定存在函数替换为key集合
     * */
    public static void replaceParagraphStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                            List<String> breaks,String separator,List<String> replaceList){
        replaceParagraphStrs(document, map, list, breaks, separator, replaceList,null,null,null,null,null);
    }

    /**
     * 替换段落中的内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * @param replaceList 使用指定存在函数替换为key集合
     * @param fontFamily 字体
     * @param fontSize 文字大小
     * @param fontBold 是否加粗
     * @param showLine 是否显示下划线
     * @param removeList 要删除标题的内容key集合
     * */
    public static void replaceParagraphStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                            List<String> breaks,String separator,List<String> replaceList,String fontFamily,Integer fontSize,
                                            Boolean fontBold,Boolean showLine,List<String> removeList){
        boolean removeTag=false;
        if(removeList!=null&&removeList.size()>0){
            removeTag=true;
        }
        boolean replaceTag=false;
        if(replaceList!=null&&replaceList.size()>0){
            replaceTag=true;
        }
        boolean flag=false;
        if(list!=null&&list.size()>0){
            flag=true;
        }
        boolean flags=false;
        if(breaks!=null&&breaks.size()>0&&separator!=null&&!"".equals(separator)){
            flags=true;
        }
        Iterator<XWPFParagraph> itPara = document.getParagraphsIterator();
        String mkey=null;
        String temp=null;
        Iterator<String> iterator = null;
        XWPFRun xwpfRun=null;
        while (itPara.hasNext()) {
            boolean tag=false;
            XWPFParagraph paragraph = (XWPFParagraph) itPara.next();
            iterator = map.keySet().iterator();
            while (iterator.hasNext()) {
                mkey = iterator.next();
                List<XWPFRun> run=paragraph.getRuns();
                for(int i=0;i<run.size();i++){
                    xwpfRun=run.get(i);
                    temp=xwpfRun.getText(xwpfRun.getTextPosition());
                    if(temp!=null ){
                        boolean tgs=false;
                        if(replaceTag&&replaceList.contains(mkey)){
                            tgs=temp.trim().contains(mkey);
                        }else{
                            tgs=temp.trim().equals(mkey);
                        }
                        if(tgs&&removeTag&&removeList.contains(mkey)){
                            xwpfRun.removeBreak();
                            continue;
                        }
                        if(tgs){
                            if(flag&&list.contains(mkey)){//设置下划线属性
                                xwpfRun.setUnderline(UnderlinePatterns.DOT_DOT_DASH);
                            }
                            if(flags&&breaks.contains(mkey)){
                                xwpfRun.setText("",0);
                                tag=true;
                                break;
                            }else{
                                if(replaceTag&&replaceList.contains(mkey)){
                                    xwpfRun.setText(replaceStrs(temp,map,replaceList),0);
                                }else{
                                    xwpfRun.setText(map.get(mkey)==null?"":map.get(mkey),0);
                                }
                                if(fontFamily!=null&&!fontFamily.equals("")){
                                    xwpfRun.setFontFamily("宋体");
                                }
                                if(fontSize!=null){
                                    xwpfRun.setFontSize(fontSize);
                                }
                                if(fontBold!=null&&fontBold){
                                    xwpfRun.setBold(true);
                                }
                                if(showLine!=null&&showLine){
                                    xwpfRun.setUnderline(UnderlinePatterns.DOT_DOT_DASH);
                                }
                            }
                        }
                    }
                }
                if(tag){
                    tag=false;
                    if(map.get(mkey)!=null){
                        String[] sts=map.get(mkey).split(separator);
                        if(sts.length>0){
                            for (int j =0; j <= sts.length-1; j++) {
                                XWPFRun run_s = paragraph.insertNewRun(paragraph.getRuns().size()-1);
                                run_s.setText(sts[j]);
                                if(j<sts.length-1){
                                    run_s.addBreak();
                                }
                                if(fontFamily!=null&&!fontFamily.equals("")){
                                    run_s.setFontFamily("宋体");
                                }
                                if(fontSize!=null){
                                    run_s.setFontSize(fontSize);
                                }
                                if(fontBold!=null&&fontBold){
                                    xwpfRun.setBold(true);
                                }
                                if(showLine!=null&&showLine){
                                    xwpfRun.setUnderline(UnderlinePatterns.DOT_DOT_DASH);
                                }
                            }
                        }
                    }
                }
            }
        }
    }

    private static String replaceStrs(String strs,Map<String,String> map,List<String> replaceList){
        String value="";
        String key="";
        for (int i = 0; i < replaceList.size(); i++) {
            key=replaceList.get(i);
            value=map.get(key);
            if(value!=null){
                strs=strs.replace(key, value);
            }
        }
        return strs;
    }

    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map){
        replaceTableStrs(document, map, null);
    }
    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map,List<String> list){
        replaceTableStrs(document, map, list, null, null);
    }
    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                        List<String> breaks,String separator){
        replaceTableStrs(document, map, list, breaks, separator,null);
    }
    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * @param removes 删除指定key所在的表格行的key集合
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                        List<String> breaks,String separator,List<String> removes){
        replaceTableStrs(document, map, list, breaks, separator, removes,null,null);
    }
    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * @param removes 删除指定key所在的表格行的key集合
     * @param tableIndex 指定的表格索引
     * @param isHidden 是否隐藏表格边框
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                        List<String> breaks,String separator,List<String> removes,Integer tableIndex,Boolean isHidden){
        replaceTableStrs(document, map, list, breaks, separator, removes, tableIndex, isHidden,null);
    }

    /**
     * 替换表格中内容
     * @param document 文档对象
     * @param map 数据集合
     * @param list 显示下划线的key集合
     * @param breaks 多行显示的key集合
     * @param separator 分隔符
     * @param removes 删除指定key所在的表格行的key集合
     * @param tableIndex 指定的表格索引
     * @param isHidden 是否隐藏表格边框
     * @param boldList 要加粗的内容集合
     * */
    public static void replaceTableStrs(XWPFDocument document,Map<String,String> map,List<String> list,
                                        List<String> breaks,String separator,List<String> removes,Integer tableIndex,Boolean isHidden,List<String> boldList){
        if(isHidden==null){
            isHidden=false;
        }
        boolean flag=false;
        if(list!=null&&list.size()>0){
            flag=true;
        }
        boolean flags=false;
        if(breaks!=null&&breaks.size()>0&&separator!=null&&!"".equals(separator)){
            flags=true;
        }
        removeTableRow(document, removes);
        if(tableIndex!=null){
            XWPFTable table=document.getTables().get(tableIndex);
            tableControl(table, map, list, breaks, separator, removes, tableIndex, isHidden, flag, flags,boldList);
        }else{
            Iterator<XWPFTable> itTable = document.getTablesIterator();
            while (itTable.hasNext()) {
                XWPFTable table = (XWPFTable) itTable.next();
                tableControl(table, map, list, breaks, separator, removes, tableIndex, isHidden, flag, flags,boldList);
            }
        }

    }

    private static String getBoldStr(String strs,List<String> boldList){
        int count=boldList.size();
        for (int i = 0; i < count; i++) {
            if(strs.contains(boldList.get(i))){
                return boldList.get(i);
            }
        }
        return "";
    }

    private static void tableControl( XWPFTable table,Map<String,String> map,List<String> list,
                                      List<String> breaks,String separator,List<String> removes,Integer tableIndex,Boolean isHidden,
                                      boolean flag,boolean flags,List<String> boldList){
        boolean tagBold=false;
        if(boldList!=null&&boldList.size()>0){
            tagBold=true;
        }
        if(isHidden){
            controlTableBorder(table);
        }
        String mkey=null;
        String mvalue=null;
        String temp=null;
        XWPFRun xwpfRun=null;
        Iterator<String> iterator = null;
        int rcount = table.getNumberOfRows();
        for (int i = 0; i < rcount; i++) {
            XWPFTableRow row = table.getRow(i);
            List<XWPFTableCell> cells = row.getTableCells();
            for (XWPFTableCell cell : cells) {
                temp=cell.getText();
                iterator = map.keySet().iterator();
                while (iterator.hasNext()) {
                    mkey = iterator.next();
                    mvalue=map.get(mkey);
                    if(mvalue==null){
                        continue;
                    }
                    String s=temp.replace(mkey, mvalue).replace(mkey, mvalue);
                    List<XWPFParagraph> paras=cell.getParagraphs();
                    if(paras.size()>0){
                        for (XWPFParagraph para : paras) {
                            boolean tag=false;
//                            para.setAlignment(ParagraphAlignment.CENTER);
//                            para.setBorderBottom(Borders.DOUBLE);
//                            para.setBorderTop(Borders.DOUBLE);
//                            para.setBorderRight(Borders.DOUBLE);
//                            para.setBorderLeft(Borders.DOUBLE);
//                            para.setBorderBetween(Borders.SINGLE);
//                            para.setVerticalAlignment(TextAlignment.TOP);
                            List<XWPFRun> runs = para.getRuns();
                            for (int k=0; k<runs.size(); k++) {
                                xwpfRun=runs.get(k);
                                String ts=xwpfRun.getText(xwpfRun.getTextPosition());
                                if(ts!=null && ts.trim().equals(mkey)){
                                    if(flag&&list.contains(mkey)){//设置下划线属性
//                                          xwpfRun.setUnderline(UnderlinePatterns.DOT_DOT_DASH);
                                        xwpfRun.setUnderline(UnderlinePatterns.SINGLE);
                                    }
                                    if(flags&&breaks.contains(mkey)){
                                        xwpfRun.setText("",0);
                                        tag=true;
                                        break;
                                    }else{
                                        String tempValue=map.get(mkey);
                                        if(tagBold){
                                            String boldStr=getBoldStr(tempValue, boldList);
                                            if(!boldStr.equals("")){
                                                int a_1=tempValue.indexOf(boldStr);
                                                String temps=tempValue.substring(a_1+boldStr.length());
                                                xwpfRun.setBold(true);
                                                xwpfRun.setFontFamily("Times New Roman");
                                                xwpfRun.setFontSize(10);
                                                xwpfRun.setText(boldStr, 0);
                                                XWPFRun xwpf=para.createRun();
                                                xwpf.setText(temps, 0);
                                            }else{
                                                xwpfRun.setText(tempValue,0);
                                            }
                                        }else{
                                            xwpfRun.setText(tempValue,0);
                                        }
                                    }
                                }
                            }
                            if(tag){
                                tag=false;
                                String[] sts=map.get(mkey).split(separator);
                                for (int j =0; j <= sts.length-1; j++) {
                                    XWPFRun run = para.insertNewRun(para.getRuns().size()-1);
//                                     run.setText(sts[j]);
                                    String tempValue=sts[j];
                                    if(tagBold){
                                        String boldStr=getBoldStr(tempValue, boldList);
                                        if(!boldStr.equals("")){
                                            int a_1=tempValue.indexOf(boldStr);
                                            String temps=tempValue.substring(a_1+boldStr.length());
                                            run.setBold(true);
                                            xwpfRun.setFontFamily("Times New Roman");
                                            xwpfRun.setFontSize(10);
                                            xwpfRun.setText(boldStr, 0);
                                            run.setText(boldStr);
                                            run=para.insertNewRun(para.getRuns().size()-1);
                                            run.setText(temps);
                                        }else{
                                            run.setText(tempValue);
                                        }
                                    }else{
                                        run.setText(tempValue);
                                    }

                                    if(j<sts.length-1){
                                        run.addBreak();
                                    }

//                                       run.setFontFamily("宋体");
//                                       run.setBold(true);
                                }
                            }
                        }
                    }else{
                        cell.setText(s);
                    }
                }
            }
        }
    }


    private static void controlTableBorder(XWPFTable ss){
        CTTblBorders borders=ss.getCTTbl().getTblPr().addNewTblBorders();
        CTBorder hBorder=borders.addNewInsideH();
        hBorder.setVal(STBorder.Enum.forString("none"));
        hBorder.setSz(new BigInteger("1"));
        hBorder.setColor("0000FF");
        CTBorder vBorder=borders.addNewInsideV();
        vBorder.setVal(STBorder.Enum.forString("none"));
        vBorder.setSz(new BigInteger("1"));
        vBorder.setColor("00FF00");
        CTBorder lBorder=borders.addNewLeft();
        lBorder.setVal(STBorder.Enum.forString("none"));
        lBorder.setSz(new BigInteger("1"));
        lBorder.setColor("3399FF");
        CTBorder rBorder=borders.addNewRight();
        rBorder.setVal(STBorder.Enum.forString("none"));
        rBorder.setSz(new BigInteger("1"));
        rBorder.setColor("F2B11F");
        CTBorder tBorder=borders.addNewTop();
        tBorder.setVal(STBorder.Enum.forString("none"));
        tBorder.setSz(new BigInteger("1"));
        tBorder.setColor("C3599D");
        CTBorder bBorder=borders.addNewBottom();
        bBorder.setVal(STBorder.Enum.forString("none"));
        bBorder.setSz(new BigInteger("1"));
        bBorder.setColor("F7E415");
    }

    /**
     * 在指定表格中增加行
     * @param document 文档对象
     * @param tableIndex 表格索引
     * @param rowIndex 行索引
     * @param datas 行数据集合
     * */
    public static void addTableRow(XWPFDocument document,int tableIndex,Integer rowIndex,List<List<String>> datas){
        addTableRow(document, tableIndex, rowIndex, datas,0);
    }
    /**
     * 在指定表格中增加行
     * @param document 文档对象
     * @param tableIndex 表格索引
     * @param rowIndex 行索引
     * @param datas 行数据集合
     * @param getRowIndex 获取指定行的列数据
     * */
    public static void addTableRow(XWPFDocument document,int tableIndex,Integer rowIndex,List<List<String>> datas,int getRowIndex){
        addTableRow(document, tableIndex, rowIndex, datas, getRowIndex, null,null,null,null);
    }
    /**
     * 在指定表格中增加行
     * @param document 文档对象
     * @param tableIndex 表格索引
     * @param rowIndex 行索引
     * @param datas 行数据集合
     * @param getRowIndex 获取指定行的列数据
     * @param align 布局方式
     * @param num 次数
     * @param fontFamily 字体
     * @param fontSize 大小
     * */
    public static void addTableRow(XWPFDocument document,int tableIndex,Integer rowIndex,List<List<String>> datas,
                                   int getRowIndex,String align,Integer num,String fontFamily,Integer fontSize){
        if(align==null){
            align="center";
        }
        boolean tag=false;
        if(num!=null){
            tag=true;
        }
        List<XWPFTable> ts=document.getTables();
        XWPFTable ss=ts.get(tableIndex);
        int c1=datas.size();
        List<String> list=null;
        XWPFTableRow rw=null;
        for (int i = c1-1; i >=0; i--) {
            list=datas.get(i);
            if(rowIndex==null){
                rowIndex=ss.getRows().size();
            }
            XWPFTableRow rws=ss.getRow(getRowIndex);
            rw=ss.insertNewTableRow(rowIndex);
            List<XWPFTableCell> rss=rws.getTableCells();
            XWPFTableCell rs=null;
            BigInteger width=BigInteger.valueOf(20);
            for (int j = 0; j < rss.size(); j++) {
                width= (BigInteger) rss.get(j).getCTTc().getTcPr().getTcW().getW();
                rs=rw.addNewTableCell();
                CTTcPr cellPr = rs.getCTTc().addNewTcPr();
                cellPr.addNewVAlign().setVal(STVerticalJc.CENTER);
                if(align.equals("center")){
                }else if(align.equals("left")){
                    cellPr.addNewVAlign().setVal(STVerticalJc.TOP);
                }else if(align.equals("right")){
                    cellPr.addNewVAlign().setVal(STVerticalJc.CENTER);
                }
                //设置宽度
                cellPr.addNewTcW().setW(width);
                XWPFParagraph p=null;
                if(rs.getParagraphs().size()==0){
                    p=rs.addParagraph();
                }else{
                    p=rs.getParagraphs().get(0);
                }
                if(align.equals("center")){
                    p.setAlignment(ParagraphAlignment.CENTER);
                }else if(align.equals("left")){
                    p.setVerticalAlignment(TextAlignment.TOP);
                    p.setAlignment(ParagraphAlignment.LEFT);
                }else if(align.equals("right")){
                    p.setAlignment(ParagraphAlignment.RIGHT);
                }
                XWPFRun r= p.createRun();
                if(fontFamily!=null&&!"".equals(fontFamily)){
                    r.setFontFamily(fontFamily);
                }
                if(fontSize!=null&&fontSize>0){
                    r.setFontSize(fontSize);
                }
                if(!tag){
                    r.setText(list.get(j));
                }
                if(tag){
                    r.setText(list.get(j));
                    for (int k = 0; k < num; k++) {
                        r.addBreak();
                    }
                }
            }
        }
    }



    /**
     * 删除表格行
     * @param document 文档对象
     * @param  key集合(存在删除行中的key)
     * */
    public static void removeTableRow(XWPFDocument document,List<String> list){
        if(list==null||list.size()==0){
            return;
        }
        String temp=null;
        Iterator<XWPFTable> itTable = document.getTablesIterator();
        Set<XWPFTableRow> rrows=null;
        while (itTable.hasNext()) {
            boolean tag=false;
            XWPFTable table = (XWPFTable) itTable.next();
            int rcount = table.getNumberOfRows();
            rrows=new HashSet<XWPFTableRow>();
            for (int i = 0; i < rcount; i++) {
                XWPFTableRow row = table.getRow(i);
                List<XWPFTableCell> cells = row.getTableCells();
                for (XWPFTableCell cell : cells) {
                    temp=cell.getText();
                    for (String string : list) {
                        if(temp.contains(string)){
                            rrows.add(row);
                            tag=true;
                        }
                    }
                }
            }
            if(tag){
                iterationTable(table, rrows);
            }
        }
    }
    private static void iterationTable(XWPFTable table,Set<XWPFTableRow> rows ){
        boolean tag=false;
        int rcount = table.getNumberOfRows();
        for (int i = 0; i < rcount; i++) {
            XWPFTableRow row = table.getRow(i);
            for (XWPFTableRow xwpfTableRow : rows) {
                if(row.equals(xwpfTableRow)){
                    table.removeRow(i);
                    rows.remove(xwpfTableRow);
                    tag=true;
                    break;
                }
            }
            if(tag){
                break;
            }
        }
        if(tag){
            iterationTable(table, rows);
        }
    }



    /**
     * 替换所有段落中的标记
     *
     * @param xwpfParagraphList
     * @param params
     */
    public static void replaceInAllParagraphs(List<XWPFParagraph> xwpfParagraphList, Map<String, String> params) {
        for (XWPFParagraph paragraph : xwpfParagraphList) {
            if (paragraph.getText() == null || paragraph.getText().equals("")) continue;
            for (String key : params.keySet()) {
                if (paragraph.getText().contains(key)) {
                    replaceInParagraph(paragraph, key, params.get(key));
                }
            }
        }
    }

    /**
     * 替换段落中的字符串
     *
     * @param xwpfParagraph
     * @param oldString
     * @param newString
     */
    public static void replaceInParagraph(XWPFParagraph xwpfParagraph, String oldString, String newString) {
        Map<String, Integer> pos_map = findSubRunPosInParagraph(xwpfParagraph, oldString);
        if (pos_map != null) {
           // System.out.println("start_pos:" + pos_map.get("start_pos"));
            // System.out.println("end_pos:" + pos_map.get("end_pos"));

            List<XWPFRun> runs = xwpfParagraph.getRuns();
            XWPFRun modelRun = runs.get(pos_map.get("end_pos"));
            XWPFRun xwpfRun = xwpfParagraph.insertNewRun(pos_map.get("end_pos") + 1);
            xwpfRun.setText(newString);
           // System.out.println("字体大小:" + modelRun.getFontSize());
            if (modelRun.getFontSize() != -1) xwpfRun.setFontSize(modelRun.getFontSize());//默认值是五号字体,但五号字体getFontSize()时,返回-1
            xwpfRun.setFontFamily(modelRun.getFontFamily());
            for (int i = pos_map.get("end_pos"); i >= pos_map.get("start_pos"); i--) {
               // System.out.println("remove run pos in :" + i);
                xwpfParagraph.removeRun(i);
            }
        }
    }


    /**
     * 找到段落中子串的起始XWPFRun下标和终止XWPFRun的下标
     *
     * @param xwpfParagraph
     * @param substring
     * @return
     */
    public static Map<String, Integer> findSubRunPosInParagraph(XWPFParagraph xwpfParagraph, String substring) {

        List<XWPFRun> runs = xwpfParagraph.getRuns();
        int start_pos = 0;
        int end_pos = 0;
        String subtemp = "";
        for (int i = 0; i < runs.size(); i++) {
            subtemp = "";
            start_pos = i;
            for (int j = i; j < runs.size(); j++) {
                if (runs.get(j).getText(runs.get(j).getTextPosition()) == null) continue;
                subtemp += runs.get(j).getText(runs.get(j).getTextPosition());
                if (subtemp.equals(substring)) {
                    end_pos = j;
                    Map<String, Integer> map = new HashMap<>();
                    map.put("start_pos", start_pos);
                    map.put("end_pos", end_pos);
                    return map;
                }
            }
        }
        return null;
    }

}
  1. FileUtils帮助类
package com.tdsms.common.utils.file;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import com.tdsms.common.utils.uuid.IdUtils;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.ArrayUtils;
import com.tdsms.common.config.RuoYiConfig;
import com.tdsms.common.utils.DateUtils;
import com.tdsms.common.utils.StringUtils;
import org.apache.commons.io.FilenameUtils;

/**
 * 文件处理工具类
 *
 * @author
 */
public class FileUtils
{
    public static String FILENAME_PATTERN = "[a-zA-Z0-9_\\-\\|\\.\\u4e00-\\u9fa5]+";

    /**
     * 输出指定文件的byte数组
     *
     * @param filePath 文件路径
     * @param os 输出流
     * @return
     */
    public static void writeBytes(String filePath, OutputStream os) throws IOException
    {
        FileInputStream fis = null;
        try
        {
            File file = new File(filePath);
            if (!file.exists())
            {
                throw new FileNotFoundException(filePath);
            }
            fis = new FileInputStream(file);
            byte[] b = new byte[1024];
            int length;
            while ((length = fis.read(b)) > 0)
            {
                os.write(b, 0, length);
            }
        }
        catch (IOException e)
        {
            throw e;
        }
        finally
        {
            IOUtils.close(os);
            IOUtils.close(fis);
        }
    }

    /**
     * 输出指定文件的byte数组
     *
     * @param file 文件路径
     * @param os 输出流
     * @return
     */
    public static void writeBytes(File file, OutputStream os) throws IOException
    {
        FileInputStream fis = null;
        try
        {
            if (!file.exists())
            {
                throw new FileNotFoundException("pdf不存在");
            }
            fis = new FileInputStream(file);
            byte[] b = new byte[1024];
            int length;
            while ((length = fis.read(b)) > 0)
            {
                os.write(b, 0, length);
            }
        }
        catch (IOException e)
        {
            throw e;
        }
        finally
        {
            IOUtils.close(os);
            IOUtils.close(fis);
        }
    }

    /**
     * 写数据到文件中
     *
     * @param data 数据
     * @return 目标文件
     * @throws IOException IO异常
     */
    public static String writeImportBytes(byte[] data) throws IOException
    {
        return writeBytes(data, RuoYiConfig.getImportPath());
    }

    /**
     * 写数据到文件中
     *
     * @param data 数据
     * @param uploadDir 目标文件
     * @return 目标文件
     * @throws IOException IO异常
     */
    public static String writeBytes(byte[] data, String uploadDir) throws IOException
    {
        FileOutputStream fos = null;
        String pathName = "";
        try
        {
            String extension = getFileExtendName(data);
            pathName = DateUtils.datePath() + "/" + IdUtils.fastUUID() + "." + extension;
            File file = FileUploadUtils.getAbsoluteFile(uploadDir, pathName);
            fos = new FileOutputStream(file);
            fos.write(data);
        }
        finally
        {
            IOUtils.close(fos);
        }
        return FileUploadUtils.getPathFileName(uploadDir, pathName);
    }

    /**
     * 删除文件
     *
     * @param filePath 文件
     * @return
     */
    public static boolean deleteFile(String filePath)
    {
        boolean flag = false;
        File file = new File(filePath);
        // 路径为文件且不为空则进行删除
        if (file.isFile() && file.exists())
        {
            file.delete();
            flag = true;
        }
        return flag;
    }

    /**
     * 文件名称验证
     *
     * @param filename 文件名称
     * @return true 正常 false 非法
     */
    public static boolean isValidFilename(String filename)
    {
        return filename.matches(FILENAME_PATTERN);
    }

    /**
     * 检查文件是否可下载
     *
     * @param resource 需要下载的文件
     * @return true 正常 false 非法
     */
    public static boolean checkAllowDownload(String resource)
    {
        // 禁止目录上跳级别
        if (StringUtils.contains(resource, ".."))
        {
            return false;
        }

        // 检查允许下载的文件规则
        if (ArrayUtils.contains(MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION, FileTypeUtils.getFileType(resource)))
        {
            return true;
        }

        // 不在允许下载的文件规则
        return false;
    }

    /**
     * 下载文件名重新编码
     *
     * @param request 请求对象
     * @param fileName 文件名
     * @return 编码后的文件名
     */
    public static String setFileDownloadHeader(HttpServletRequest request, String fileName) throws UnsupportedEncodingException
    {
        final String agent = request.getHeader("USER-AGENT");
        String filename = fileName;
        if (agent.contains("MSIE"))
        {
            // IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        }
        else if (agent.contains("Firefox"))
        {
            // 火狐浏览器
            filename = new String(fileName.getBytes(), "ISO8859-1");
        }
        else if (agent.contains("Chrome"))
        {
            // google浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        else
        {
            // 其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        return filename;
    }

    /**
     * 下载文件名重新编码
     *
     * @param response 响应对象
     * @param realFileName 真实文件名
     */
    public static void setAttachmentResponseHeader(HttpServletResponse response, String realFileName) throws UnsupportedEncodingException
    {
        String percentEncodedFileName = percentEncode(realFileName);

        StringBuilder contentDispositionValue = new StringBuilder();
        contentDispositionValue.append("attachment; filename=")
                .append(percentEncodedFileName)
                .append(";")
                .append("filename*=")
                .append("utf-8''")
                .append(percentEncodedFileName);

        response.addHeader("Access-Control-Expose-Headers", "Content-Disposition,download-filename");
        response.setHeader("Content-disposition", contentDispositionValue.toString());
        response.setHeader("download-filename", percentEncodedFileName);
    }

    /**
     * 百分号编码工具方法
     *
     * @param s 需要百分号编码的字符串
     * @return 百分号编码后的字符串
     */
    public static String percentEncode(String s) throws UnsupportedEncodingException
    {
        String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString());
        return encode.replaceAll("\\+", "%20");
    }

    /**
     * 获取图像后缀
     *
     * @param photoByte 图像数据
     * @return 后缀名
     */
    public static String getFileExtendName(byte[] photoByte)
    {
        String strFileExtendName = "jpg";
        if ((photoByte[0] == 71) && (photoByte[1] == 73) && (photoByte[2] == 70) && (photoByte[3] == 56)
                && ((photoByte[4] == 55) || (photoByte[4] == 57)) && (photoByte[5] == 97))
        {
            strFileExtendName = "gif";
        }
        else if ((photoByte[6] == 74) && (photoByte[7] == 70) && (photoByte[8] == 73) && (photoByte[9] == 70))
        {
            strFileExtendName = "jpg";
        }
        else if ((photoByte[0] == 66) && (photoByte[1] == 77))
        {
            strFileExtendName = "bmp";
        }
        else if ((photoByte[1] == 80) && (photoByte[2] == 78) && (photoByte[3] == 71))
        {
            strFileExtendName = "png";
        }
        return strFileExtendName;
    }

    /**
     * 获取文件名称 /profile/upload/2022/04/16/tt.png -- tt.png
     *
     * @param fileName 路径名称
     * @return 没有文件路径的名称
     */
    public static String getName(String fileName)
    {
        if (fileName == null)
        {
            return null;
        }
        int lastUnixPos = fileName.lastIndexOf('/');
        int lastWindowsPos = fileName.lastIndexOf('\\');
        int index = Math.max(lastUnixPos, lastWindowsPos);
        return fileName.substring(index + 1);
    }

    /**
     * 获取不带后缀文件名称 /profile/upload/2022/04/16/tt.png -- tt
     *
     * @param fileName 路径名称
     * @return 没有文件路径和后缀的名称
     */
    public static String getNameNotSuffix(String fileName)
    {
        if (fileName == null)
        {
            return null;
        }
        String baseName = FilenameUtils.getBaseName(fileName);
        return baseName;
    }

}
  1. pdf生成帮助类
package com.tdsms.web.controller.tool;

import com.aspose.words.Document;
import com.aspose.words.FontSettings;
import com.aspose.words.License;
import com.aspose.words.SaveFormat;
import org.apache.poi.xwpf.usermodel.XWPFDocument;

import java.io.*;

public class AsposeUtil {

    /**
     * 验证License 若不验证则转化出的pdf文档会有水印产生
     * @return
     */
    public boolean getLicense() {
        boolean result = false;
        try {
            InputStream is =this.getClass().getClassLoader().getResourceAsStream("license.xml");
            License aposeLic = new License();
            aposeLic.setLicense(is);
            result = true;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return result;
    }

    public static void main(String[] args) throws Exception {
        AsposeUtil bean = new AsposeUtil();
        bean.word2Pdf2("C:\\Users\\Rzxuser\\Downloads\\文档安全.docx","D:\\安全文档.pdf");
        // bean.word2Pdf2("http://111.47.13.71:36038/tdsms/20230112-ffaa9d61-e8da-4443-bd01-755587a1655c.docx","D:\\test11.pdf");


       /* XWPFDocument document = new XWPFDocument ();//新建文档 后面NEW方法可以忽略
        ByteArrayOutputStream baos = new ByteArrayOutputStream();//二进制OutputStream
        document.write(baos);//文档写入流
        ByteArrayInputStream in = new ByteArrayInputStream(baos.toByteArray());//OutputStream写入InputStream二进制流

        bean.word2pdf("D:\\test11",in,"test");*/
    }

    /**
     * word转pdf
     * inpath: 输入word的路径
     * outpath: 输出pdf的路径
     */
    public void word2Pdf2(String inpath,String outpath) throws Exception {
        if (!getLicense()) {
            System.out.println("非法------------");
            return;
        }

        System.out.println("开始使用Aspose.words进行转换");

        long old = System.currentTimeMillis();
        File file = new File(outpath);

        FileOutputStream os = new FileOutputStream(file);

        //解决乱码
        //如果是windows执行,不需要加这个
        String osName = System.getProperty("os.name", "");
        if (osName.startsWith("Mac OS")) {
        } else if (osName.startsWith("Windows")) {
        } else { // assume Unix or Linux
            //如果是linux执行,需要添加这个 ,如果还有乱码,可以把/usr/share/fonts/chinese路径下的所有文件拷贝到有问题的环境。并且再执行:source /etc/profile
            new FontSettings().setFontsFolder("/usr/share/fonts/chinese",true);
        }

        Document doc = new Document(inpath);

        //全面支持DOC, DOCX, OOXML, RTF HTML, OpenDocument, PDF, EPUB, XPS, SWF 相互转换
        doc.save(os, SaveFormat.PDF);
        os.close();
        long now = System.currentTimeMillis();
        System.out.println("Aspose.words转换结束,共耗时:" + ((now - old) / 1000.0) + "秒");
    }


    /**
     * word转pdf
     * @param path      pdf输出路径
     * @param wordInput word输入流
     * @param wordName  word文档的名称
     */
    public void word2pdf(String path, InputStream wordInput, String wordName) throws FileNotFoundException {
        if (!getLicense()) {
            System.out.println("非法");
            return;
        }

        //新建一个空白pdf文档
        long old = System.currentTimeMillis();
        File file = new File(path + wordName + ".pdf");
        FileOutputStream os = new FileOutputStream(file);

        //解决乱码
        //如果是windows执行,不需要加这个
        String osName = System.getProperty("os.name", "");
        if (osName.startsWith("Mac OS")) {
        } else if (osName.startsWith("Windows")) {
        } else { // assume Unix or Linux
            //如果是linux执行,需要添加这个 ,如果还有乱码,可以把/usr/share/fonts/chinese路径下的所有文件拷贝到有问题的环境。并且再执行:source /etc/profile
            new FontSettings().setFontsFolder("/usr/share/fonts/dejavu",true);
        }

        //Address是将要被转化的word文档
        Document doc = null;
        try {
            doc = new Document(wordInput);
        } catch (Exception e) {
            e.printStackTrace();
        }
        try {
            //全面支持DOC, DOCX, OOXML, RTF HTML, OpenDocument, PDF, EPUB, XPS, SWF 相互转换
            doc.save(os, SaveFormat.PDF);
        } catch (Exception e) {
            e.printStackTrace();
        }
        long now = System.currentTimeMillis();
        //转化用时
        System.out.println("共耗时:" + ((now - old) / 1000.0) + "秒");
    }
}
  1. 效果图

模板




Java poi生成Word带页码_pdf


效果


Java poi生成Word带页码_word_02


Java poi生成Word带页码_pdf_03


  1. docx转pdf需要jar和xml


Java poi生成Word带页码_Java poi生成Word带页码_04


Java poi生成Word带页码_word_05


<dependency>
            <groupId>com.aspose.words</groupId>
            <artifactId>aspose-words</artifactId>
            <version>19.5</version>
            <scope>system</scope>
            <systemPath>${project.basedir}/lib/aspose-words-19.5.jar</systemPath>
        </dependency>
 <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <version>2.1.1.RELEASE</version>
                <configuration>
                    <includeSystemScope>true</includeSystemScope>
                </configuration>
            </plugin>
  1. 注意的坑
1、将word另存为xml文件(文件类型一定选择word xml文档),在编辑器中将变量写进去,然后存为docx。
2、word 模板,第一个图表最后一个用户数据 必须2空格呀!!!不然word转pdf的时候,x轴用户姓名会不显示的!!
3、需要把winds下的字体 拷贝到linux下面去,不然生成的pdf会有乱码。()
需要的jar包和xml
链接: https://pan.baidu.com/s/1euMsnp6oG7oFHrVE75B16w 
提取码: dkde