Java pdf下载优化

之前博客中写过项目中实现pdf下载的需求(文章详见),开发、测试、以及验收的时候均没有问题,前段时间业务方找我说下载pdf的功能总是报超时错误,我排查了下原因,原来是pdf中放的图片都是8M左右的,这肯定扛不住啊。问题发现了那就解决。思路有两个:

  1. 压缩图片
  2. 异步下载

1.压缩图片

这里使用Google开源的图片压缩库thumbnailator,这个工具简单易用,支持各种压缩方式,提供链式API,用兴趣的可以去研究下。
项目引入依赖:

<dependency>
    <groupId>net.coobird</groupId>
    <artifactId>thumbnailator</artifactId>
    <version>0.4.8</version>
</dependency>

pfd渲染前,先把获取到的图片压缩,伪代码如下(源代码详见):

Lists.partition(imageUrls, BULK_SIZE).forEach(items -> {
        FraudReportOrderEvidenceImage image = new FraudReportOrderEvidenceImage();
        for (int i = 0; i < items.size(); i++) {

            byte[] imageBytes = restTemplate.getForObject(items.get(i), byte[].class);
            ByteArrayInputStream bais = new ByteArrayInputStream(imageBytes);
            ByteArrayOutputStream baos = new ByteArrayOutputStream(BAOS_SIZE);

            try {
                //压缩图片
                Thumbnails.of(bais).scale(IMAGE_SCALE_SIZE).toOutputStream(baos);
                byte[] thumbnailBytes = baos.toByteArray();
                if (i == 0) {
                    image.setReportDescImage(this.byte2String(thumbnailBytes));
                    continue;
                }
                if (i == 1) {
                    image.setReportDescImage2(this.byte2String(thumbnailBytes));
                    continue;
                }
                if (i == 2) {
                    image.setReportDescImage3(this.byte2String(thumbnailBytes));
                    continue;
                }
                if (i == 3) {
                    image.setReportDescImage4(this.byte2String(thumbnailBytes));
                }
            } catch (Exception e) {
                log.error("压缩图片失败", e);
                throw new ServiceResponseException("压缩图片失败");
            }
        }
        images.add(image);
    });

log.info("read {} images byte costs: {}ms", imageUrls.size(), System.currentTimeMillis() - start);

经过thumbnailator压缩之后,图片字节大大缩小,同时图片并没有失真模糊,在页面点击下载按钮,实时下载正常,不再超时

2.异步下载

实际上,下载文件在系统中一般都是非核心流程,所以大多数系统都是采用异步下载的方式,所谓异步下载是相对系统实时渲染数据给到前端供用户下载而言的,实现方式就是系统在业务低峰期(比如凌晨2点)通过job定时去获取指定的数据,生成相应的文件,然后上传到file server或者OSS服务器,上传成功后,系统库表中保留文件资源的地址。下载页面会把相应记录对应的文件地址返回给前端,用户下载的时候,前端直接通过地址url去download指定的文件即可,这样下载的处理就从业务系统中剥离出去,不会因为大文件的耗时处理影响业务系统性能。一般批处理系统、数仓、BI等都是以这种方式生成文件,供客户端下载。需要说明的是,此种方式下载,数据一般是T+1的,但由于是非核心功能,所以T+1是完全没问题的。

ps:
每张图片8M左右,压缩之后正常下载
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java使用com.lowagie.text.pdf插件编写的PDF报表工具类,支持动态报表创建,使用简单,附件中包含了测试类和生成的报表文件。附件中的代码需要修改相关的保存路径后可以直接使用。创建一张报表例子: private JsFileExportResult createRowReport() { String condition = "开始时间:2018-02-02 14:00:30 结束时间:2018-02-06 16:00:30"; PDFGridReport pdfReport = new PDFGridReport("报表创建测试", GridReportTestModel.getModels()); pdfReport.addCondition(condition); pdfReport.header("字段名称1", "item1", 0, 0).width(160); pdfReport.header("字段名称3", "item3", 0, 2).getCell().backgroundColor(Color.ORANGE); pdfReport.header("字段名称4", "item4", 0, 3); pdfReport.header("字段名称5", "item5", 0, 4); pdfReport.header("字段名称2", "item2", 0, 1); pdfReport.header("值", "value", 0, 5).alignH(HAlign.ALIGN_CENTER).getCell().alignH(HAlign.ALIGN_RIGHT); pdfReport.header("时间", "time", 0, 6); pdfReport.header("图片", "image", 0, 7).width(60).alignH(HAlign.ALIGN_CENTER).getCell() .alignH(HAlign.ALIGN_CENTER).alignV(VAlign.ALIGN_MIDDLE); // 横向打印 pdfReport.getPageSetting().setPrintHorizontal(); pdfReport.group("item1").childGroup("item2"); pdfReport.setCellFormat(new PDFCellValueFormat() { @Override public String format(String fieldName, String oriValue) { if ("value".equals(fieldName)) { return String.format("%.2f", Double.parseDouble(oriValue)).toString(); } else { return oriValue; } } }); JsFileExportResult fileResult = pdfReport.createReport(); return fileResult; }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值