sponsored links

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

一、序言

Excel、PDF的匯出、匯入是我們工作中經常遇到的一個問題,剛好今天公司業務遇到了這個問題,順便記個筆記以防下次遇到相同的問題而束手無策。

公司有這麼兩個需求:

需求一、給了一個表單,讓把查出來的資料組裝到表單中並且提供以PDF格式的下載功能。

需求二、 將資料查出來以Excel表格的形式下載下來。

二、Java實現PDF的生成和資料動態插入、匯出功能

1、第一步:PDF製作模板

因為PDF常用的軟體不讓支援編輯,我們就先使用WPS以Word的形式進行編輯製作出與客戶需求一樣的樣式,然後直接另存為 .pdf 的形式如下圖所示:

a.Word裡面製作模板

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

b.更改名字為 .pdf形式

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

c.這時需要用到一個叫:Adobe Acrobat DC的軟體,具體操作如下:

用Adobe Acrobat DC開啟我們剛才改過名字的PDF檔案,點選右下角的“更多工具”按鈕

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

到下面這個頁面再點選“準備表單”按鈕

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

d.接下來就需要詳細的配置你的資料來源了

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

資料來源即:你程式碼中實體類中對應的資料(注意欄位一定要一一對應),配置完畢就可以儲存進行下面的程式碼編寫工作了。

2、程式碼的編寫

假定我們實體類什麼的都已經編寫完成、資料透過前端傳入獲取、模板位置在E盤根目錄下名字為:車輛維修審批單.pdf

匯入jar包:

<!-- PDF匯出-->
<!-- https://mvnrepository.com/artifact/com.itextpdf/itextpdf -->
<dependency>
    <groupId>com.itextpdf</groupId>
    <artifactId>itextpdf</artifactId>
    <version>5.5.13</version>
</dependency>

實現生成PDF、資料插入、匯出

@RegisterToSMP(serviceDisplay = "預覽頁面PDF下載")
@RequestMapping(value = "/DM/gwclwxsq/qygl/exportPDF$m=query.service",method =RequestMethod.POST)
public String exportPdf(@RequestBody GwclwxsqBean gwclwxsqBean , HttpServletResponse response) throws UnsupportedEncodingException {
    // 1.指定解析器
    System.setProperty("javax.xml.parsers.DocumentBuilderFactory",
            "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl");
    String filename="車輛維修審批單.pdf";
    String path="e:/";
    response.setContentType("application/pdf");
    response.setHeader("Content-Disposition", "attachment;fileName="
            + URLEncoder.encode(filename, "UTF-8"));
    OutputStream os = null;
    PdfStamper ps = null;
    PdfReader reader = null;
    try {
        os = response.getOutputStream();
        // 2 讀入pdf表單
        reader = new PdfReader(path+ "/"+filename);
        // 3 根據表單生成一個新的pdf
        ps = new PdfStamper(reader, os);
        // 4 獲取pdf表單
        AcroFields form = ps.getAcroFields();
        // 5給表單新增中文字型 這裡採用系統字型。不設定的話,中文可能無法顯示
        BaseFont bf = BaseFont.createFont("C:/WINDOWS/Fonts/SIMSUN.TTC,1",
                      BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
        form.addSubstitutionFont(bf);
        // 6查詢資料================================================
        Map<String, String> data = new HashMap<String, String>();
              data.put("commitTime", gwclwxsqBean.getCommitTime());
              data.put("driver", gwclwxsqBean.getDriver());
              data.put("carId", gwclwxsqBean.getCarId());
              data.put("carType", gwclwxsqBean.getCarType());
              data.put("repairAddress", gwclwxsqBean.getRepairAddress());
              data.put("repairCost",gwclwxsqBean.getRepairCost());
              data.put("project", gwclwxsqBean.getProject());
              data.put("fwbzzxfzrYj", gwclwxsqBean.getFwbzzxfzrYj());
              data.put("fgldspYj", gwclwxsqBean.getFgldspYj());
              data.put("remarks", gwclwxsqBean.getRemarks());
         // 7遍歷data 給pdf表單表格賦值
        for (String key : data.keySet()) {
            form.setField(key,data.get(key).toString());
        }
        ps.setFormFlattening(true);
        log.info("*******************PDF匯出成功***********************");
    } catch (Exception e) {          log.error("*******************PDF匯出失敗***********************");
        e.printStackTrace();
    } finally {
        try {
            ps.close();
            reader.close();
            os.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    return null;
}

3、測試

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

三、Java實現Excel生成和資料插入、匯出

這個比較簡單,直接上程式碼(假定你的實體類、查詢什麼的都已經寫好)注意:實體類一個是你自己的資料實體類還有一個是你匯出時表格中對應的實體類。100道Java中高階面試題彙總

我們以一個真實的公司業務來舉個例子(一個統計疫情登記人員資訊的Excel匯出功能)

a.表頭對應實體類ExportYqfkdj.java

import lombok.Data;

/**
 * description:
 * @author: zhouhong
 * @version: V1.0.0
 * @date: 2021年1月14日 下午3:05:54
 */
@Data
public class ExportYqfkdj {
    /**
     * 序號
     */
    private Integer xuhao;
    /**
     * 姓名
     */
    private String xingming;
    /**
     * 證件號碼
     */
    private String zjhm;
    /**
     * 聯絡電話
     */
    private String lxdh;
    /**
     * 申請人工作單位
     */
    private String sqrGzdw;
    /**
     * 是否接觸過疑似病例
     */
    private String sfjcgysbl;
    /**
     * 當前是否與居家隔離人員同住
     */
    private String sfyjjglrytz;
    /**
     * 當前狀態
     */
    private String dqzt;
    /**
     * 當前健康狀態
     */
    private String dqjkzt;

    /**
     * 當前體溫
     */
    private String dqtw;
    /**
     * 當前所在地址
     */
    private String dqszdz;
    /**
     * 當前居住地址
     */
    private String dqjzdz;
    /**
     * 提交時間
     * */
    private String tjsj;
}

b.Service層

/**
 * 匯出
 * @param yqfkdjBean
 * @author zhouhong
 * @return
 * @throws Exception
 */
@Transactional(rollbackFor = { Exception.class })
public DataResult exporYqfkdj(YqfkdjBean yqfkdjBean) throws Exception {
    DataResult result = new DataResult();
    List<ExportYqfkdj> list = new ArrayList<ExportYqfkdj>();
    try {
        /* 查詢匯出資訊 */
        result = getYqfkMhCXQuery(yqfkdjBean);
        SimpleDateFormat df = new SimpleDateFormat("yyyyMMddhhmmssSSS");
        for (int i = 0; i < result.getTotalcount(); i++) {
            ExportYqfkdj dmKhfwdcDtjlZxDto = new ExportYqfkdj();
            dmKhfwdcDtjlZxDto = ObjectUtil.parsePojo(result.getResults().get(i), ExportYqfkdj.class);
            dmKhfwdcDtjlZxDto.setXuhao(i + 1);
            list.add(dmKhfwdcDtjlZxDto);
        }
        String filepath = "D:/疫情防控資訊" + df.format(new Date()) + ".xlsx";
        if (System.getProperty(YqfkdjUtils.Wjdz.NAME).toLowerCase().startsWith(YqfkdjUtils.Wjdz.LI)
                || System.getProperty(YqfkdjUtils.Wjdz.NAME).toLowerCase().startsWith(YqfkdjUtils.Wjdz.LIN)) {
            filepath = "/home/Tomcat/temp/" + df.format(new Date()) + ".xlsx";
        }
        EasyExcel.write(filepath, ExportYqfkdj.class).head(head()).sheet().doWrite(list);
        result.setResults(list);
        result.setSuccess(true);
        result.setMsg(filepath);
    } catch (Exception e) {
        result.setSuccess(false);
        result.setMsg(YqfkdjUtils.Cytx.DCSB);
        e.printStackTrace();
        throw e;
    }
    return result;
}
/**
 * 疫情防控資訊匯出表頭
 * @author zhouhong
 * @return List<List<String>>
 */
private List<List<String>> head() {
    List<List<String>> list = new ArrayList<List<String>>();
    List<String> head0 = new ArrayList<String>();
    head0.add("序號");
    List<String> head1 = new ArrayList<String>();
    head1.add("姓名");
    List<String> head2 = new ArrayList<String>();
    head2.add("證件號碼");
    List<String> head3 = new ArrayList<String>();
    head3.add("聯絡電話");
    List<String> head4 = new ArrayList<String>();
    head4.add("工作所在單位");
    List<String> head5 = new ArrayList<String>();
    head5.add("是否接觸疑似病例");
    List<String> head6 = new ArrayList<String>();
    head6.add("是否與隔離人員同住");
    List<String> head7 = new ArrayList<String>();
    head7.add("當前狀態");
    List<String> head8 = new ArrayList<String>();
    head8.add("當前健康狀態");
    List<String> head9 = new ArrayList<String>();
    head9.add("體溫(°C)");
    List<String> head10 = new ArrayList<String>();
    head10.add("當前所在地址");
    List<String> head11 = new ArrayList<String>();
    head11.add("當前居住地址");
    List<String> head12 = new ArrayList<String>();
    head12.add("提交時間");
    list.add(head0);
    list.add(head1);
    list.add(head2);
    list.add(head3);
    list.add(head4);
    list.add(head5);
    list.add(head6);
    list.add(head7);
    list.add(head8);
    list.add(head9);
    list.add(head10);
    list.add(head11);
    list.add(head12);
    return list;
}

c.Controller層

@RegisterToSMP(serviceDisplay = "疫情防控查詢匯出")
@RequestMapping(value = "/DM/yqfkdj/gr/yqfkdjdc$m=export.service", method = RequestMethod.POST)
public void exportKhfxxx(@RequestBody YqfkdjBean yqfkdjBean, HttpServletResponse resp) throws Exception {
    DataResult result = new DataResult();
    try {
        SimpleDateFormat df = new SimpleDateFormat("yyyyMMddhhmmssSSS");
        result = yqfkdjService.exporYqfkdj(yqfkdjBean);
        String filepath = result.getMsg().replace("\"", "");
        File file = new File(filepath);
        String filename = "疫情防控資訊" + df.format(new Date()) + ".xlsx";
        InputStream fis = new BufferedInputStream(new FileInputStream(filepath));
        byte[] buffer = new byte[fis.available()];
        fis.read(buffer);
        fis.close();
        resp.reset();
        resp.setHeader("Content-Disposition",
                "attachment;filename=" + new String(filename.replaceAll(" ", "").getBytes("gbk")));
        resp.setHeader("Content-Length", "" + file.length());
        OutputStream os = new BufferedOutputStream(resp.getOutputStream());
        resp.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
        // 輸出檔案
        os.write(buffer);
        os.flush();
        os.close();
    } catch (Exception e) {
        e.printStackTrace();
        log.info(YqfkdjUtils.Cytx.DCSB);
        throw e;
    }
}

測試

接了個變態需求:生成 Excel + PDF 匯出,用 Java 怎麼實現?

已經全部完成PDF和Excel的生成、插入、匯出功能。

作者:慢時光
來源:cnblogs.com/Tom-shushu/p/14279357.html

分類: 數碼
時間: 2021-11-22

相關文章

6400萬超清四攝 7.46mm輕薄設計 榮耀暢玩20 Pro售價1699元

6400萬超清四攝 7.46mm輕薄設計 榮耀暢玩20 Pro售價1699元
9月17日,榮耀暢玩系列釋出了一款新品榮耀暢玩20 Pro,售價1699元.從配置上看,6400萬畫素超清四攝.6.53英寸廣色域OLED真彩屏.22.5W超級快充,可以說拍照.螢幕.快充面面俱到,整 ...

新款MacBook Pro曝光,這9款寶藏級軟體,近期值得體驗
蘋果又要開發佈會了,新一代的MacBook Pro已經曝光,如果你打算入手Mac,下面這9款寶藏級軟體,近期值得體驗. 01^無痕去水印(消除水印) 平時我們下載的圖片或影片含有水印,真是讓人頭疼,所 ...

定檔十月份?華為Mate 50 Pro曝光,外觀意料之中

定檔十月份?華為Mate 50 Pro曝光,外觀意料之中
iPhone 13的釋出,不僅使得蘋果在售的老款iPhone有了不同幅度的降價,還使得安卓陣營中的各個廠商顯得有些尷尬,畢竟此次新款iPhone在全球缺芯和疫情的影響下,並沒有外界傳言的漲價,反而進行 ...

2022 年款iPad Pro曝光,iPad 8跌至白菜價悲慘讓路

2022 年款iPad Pro曝光,iPad 8跌至白菜價悲慘讓路
9 月20 日訊息彭博社的馬克古爾曼(Mark Gurman) 在他最新一期的時事通訊<Power On>分享中表達了他對未來蘋果產品的期望. Gullman 預計新款MacBook Pr ...

華為Mate50 Pro曝光,升級後確實會變得很優秀?

華為Mate50 Pro曝光,升級後確實會變得很優秀?
加速度獨家報道:近日,網路上曝光了一張預熱海報,它顯示華為在10月21日或有一場全球釋出會,地點在奧地利維也納,有業內指出這可能就是華為Mate 50 Pro高階旗艦新機.再結合目前傳來的外觀設計和配 ...

驍龍888 Plus+屏下前攝?中興新品官宣,全新面板加持

驍龍888 Plus+屏下前攝?中興新品官宣,全新面板加持
雖然前段時間的iPhone 13加量還降價的操作,讓網友直呼真香,但是都到了2021年,還堅持使用劉海設計,遭到了不少人的詬病.從未來趨勢來看,提升屏佔比最好的解決方案還是使用屏下前攝方案. 隨著屏下 ...

華為Mate50 Pro曝光:居中挖孔+超曲面屏

華為Mate50 Pro曝光:居中挖孔+超曲面屏
根據外媒透露,華為Mate50 Pro已經曝光了,該機仍然採用超曲面屏設計,6.8英寸OLED螢幕,不過不再是左側挖孔雙攝,而是居中挖孔方案. 華為Mate 50 Pro或將還有一塊小副屏,呈豎置長方 ...

疑似小米Civi Pro曝光,延續經典設計,螢幕解析度高達4K

疑似小米Civi Pro曝光,延續經典設計,螢幕解析度高達4K
今天,一張疑似小米Civi Pro的證件照被曝光,螢幕的解析度為3840X2160,達到了4K級別.並且,這塊螢幕的尺寸為6.55英寸,正面屏佔比看起來非常高.它的外觀延續了小米civi的經典設計風格 ...

新MacBook Pro曝光:M1X加持 SD卡槽迴歸
訊息稱蘋果將於10月份再度舉辦釋出會帶來新款MacBookPro機型,而近日訊息人士@MajinBuOfficial也曝光了這款採用M1X晶片的MacBook Pro的原型機實拍. 據@MajinBu ...

華為Mate 50 Pro曝光

華為Mate 50 Pro曝光
華為P50系列手機發布還不到三個月,此前很多觀點認為,Mate數字旗艦今年不會更新了,恐怕要等到明年. 但來自螢幕供應鏈分析機構DSCC的高階專家David Naranjo分享了一份最新文件,指出Ma ...

Redmi K50 Pro曝光!驍龍898+6400萬三攝

Redmi K50 Pro曝光!驍龍898+6400萬三攝
中關村線上訊息:Redmi K50 系列即將釋出,而現在又有最新爆料釋出. 從大神對MIUI程式碼的挖掘結果來看,Redmi K50 Pro 將搭載驍龍 898,代號為"Ingres&quo ...

OPPO Reno7 Pro曝光,頂級處理器+5200mAh大電池加持,值得期待

OPPO Reno7 Pro曝光,頂級處理器+5200mAh大電池加持,值得期待
主要針對年輕使用者群體的OPPO Reno系列定位十分清晰,從外觀設計到配置,再到各方面的體驗都深受年輕使用者喜愛.作為快銷產品,OPPO Reno系列的更新速度也非常快,近日網上就傳出了OPPO R ...

20萬左右第五代途勝L正面挑戰同級合資SUV-皓影、榮放、途嶽

20萬左右第五代途勝L正面挑戰同級合資SUV-皓影、榮放、途嶽
在20萬左右的合資品牌SUV當中,合資品牌迎來一個新的夥伴-北京現代第五代途勝L,這個車在20萬這個級別的表現怎麼樣?我們從外觀尺寸.動力.保養幾個方面來帶您瞭解一下北京現代第五代途勝L. 1. 外觀 ...

iPhone13 Pro強得很!談談手機如何取代相機成為Vlog影片最強裝置

iPhone13 Pro強得很!談談手機如何取代相機成為Vlog影片最強裝置
前言 哈嘍,又是我,歡迎回來「Geek7even<極物研習社>」 最近這幾年,手機拍照是越來越強,各家品牌輪番交替拍照第一的寶座.但同樣是這顆攝像頭拍攝的景象,這顆CMOS採集的光線,這顆 ...

第二次購買南卡 NANK Lite Pro半入耳藍芽耳機的理由

第二次購買南卡 NANK Lite Pro半入耳藍芽耳機的理由
南卡 NANK Lite Pro半入耳藍芽耳機推出已經有段時間了,上次給大家介紹了白色版本,這次再次給大家推薦,這也是我的第二個南卡 NANK Lite Pro半入耳藍芽耳機.為什麼第二次購買了?因為 ...

瘋狂壓榨6小時!iQOO Z5電量依舊未到警戒線

瘋狂壓榨6小時!iQOO Z5電量依舊未到警戒線
現如今的手機發展之迅速,讓人們不得不為之驚歎,而我們能夠很明顯地看到,那些更貴的"旗艦手機",更多時候是把重心放在了影像系統,更多的噱頭都來自於那一塊說小不小.說大不大的鏡頭模組, ...

全球限量中國風手電!顏值效能兼備的傲雷指揮家3龍鳳限量版來了

全球限量中國風手電!顏值效能兼備的傲雷指揮家3龍鳳限量版來了
這些年買過不少手電,畢竟在生活.工作上都需要用到,有時去戶外露營更是少不了手電.前幾天夜裡家裡的空氣開關壞了,多虧有手電,提供了持續的照明才得以快速修復:工作上用到的頻率更高,由於建築工地是我的戰場, ...

出門不再帶線:倍思布洛克帶線數顯快充移動電源

出門不再帶線:倍思布洛克帶線數顯快充移動電源
如果出遠門,你的包包裡必帶的什麼東西呢? 手機,化妝包,充電器,證件,又或者還有藍芽耳機? 好像還有一個東西沒有提到,移動電源. 這個在幾年前看似可有可無的配件,在幾年的時間裡迅速成了我們所有人出行必 ...

手把手教你,如何解鎖 戴爾G15 RTX 3060 130W功率+獨顯直連

手把手教你,如何解鎖 戴爾G15 RTX 3060 130W功率+獨顯直連
前言 2021年選一款遊戲本有兩點非常重要:一個是看顯示卡的功耗.另一個是看有沒有獨顯直連.因為這兩點會直接影響筆記本效能,尤其是遊戲效能.在7月份的時候,曾經給表弟推薦了戴爾 DELL 遊匣 G15 ...

iPhone13快充會發燙?浦記這條MFi快充線堪稱最佳方案

iPhone13快充會發燙?浦記這條MFi快充線堪稱最佳方案
iPhone13釋出後,充電器的選擇十分多,但線材一開始只有原廠蘋果C91 USB-C to Lightning可選,價格並不友好.自從蘋果給第三方品牌開放C94 USB-C to Lightning ...