【精】【PDF链接转图片】- Java用pdfbox将PDF的URL转换并压缩成图片,解决“口口口”乱码问题

【业务场景】:做一个开电子发票的业务,中税返回我们一个pdf的url,这个url在web端是可以显示的,移动端:ios可以正常显示,安卓显示为是否要下载。产品邀请发票预览需让用户第一时间看到,不应该有下载的场景出现。

【解决方案】:将PDF转化图片流,并用 Base64 做加密传给前台,在此过程中很顺利,但到最后作战时的时候新的问题出现了:url转成图片之后过大,前端无法解密作展示,无奈只好后台给予处理,将图片进行压缩。具体代码如下:

  1、导入依赖:pdfbox.jar

        <!--依赖jar pdf转图片工具 -->
        <dependency>
            <groupId>org.apache.pdfbox</groupId>
            <artifactId>pdfbox</artifactId>
            <version>2.0.13</version>
        </dependency>

  2、代码实现(只写实现):

        String str = ""; // 返回给前台的Base64加密字符串
        InputStream inputStream = null;
        FileInputStream fis = null;
        ByteArrayOutputStream bos = null;
try {
                     System.setProperty("sun.java2d.cmm", "sun.java2d.cmm.kcms.KcmsServiceProvider");
                    // 1、请求pdf的url,获取文件输入流
                    String path = new String(pdfUrl.getBytes("iso8859-1"), "utf-8");
                    URL url = new URL(path);
                    inputStream = url.openStream();
                    // 2、加载pdf输入流
                    PDDocument doc = PDDocument.load(inputStream);
                    PDFRenderer renderer = new PDFRenderer(doc);
                    // 3、获取pdf的页数
                    int pageCount = doc.getNumberOfPages();
                    byte[] data = null;
                    // 4、循环页数将pdf缩放,并转为图片
                    for (int i = 0; i < pageCount; i++) {
                        // 5、方式1-第二个参数是设置缩放比(即像素),方式2-renderer.renderImageWithDPI(i, 10)
                        BufferedImage image = renderer.renderImage(i, 1.8f);
                        // 6、设置文件存放地址(这边临时存放在当前文件下,以id为文件名)
                        File file = new File(System.getProperty("user.dir") + "/" + id + ".jpg");
                        // 7、将图片BufferedImage转为png格式的图片并写入7步骤的文件中
                        ImageIO.write(image, "jpg", file);
                        // 8、将文件流转为数组
                        fis = new FileInputStream(file);
                        bos = new ByteArrayOutputStream(1000);
                        byte[] b = new byte[1000];
                        int n;
                        while ((n = fis.read(b)) != -1) {
                            bos.write(b, 0, n);
                        }
                        inputStream.close();
                        data = bos.toByteArray();
                        // 9、将文件数组用Base64加密返回字符串给前端
                        str = new String(Base64.encodeBase64String(data));
                        // 10、删除当前目录下的该文件
                        file.delete();
                    }
                } catch (MalformedURLException e) {
                    e.printStackTrace();
                } catch (IOException e) {
                    e.printStackTrace();
                } finally {
                    try {
                        if (inputStream != null) {
                            inputStream.close();
                        }
                        if (fis != null) {
                            fis.close();
                        }
                        if (bos != null) {
                            bos.close();
                        }
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }

3、图片生成后出现“口口口”的问题原因:

当前系统所在服务器上没有对应的字体,具体警告如下:

 生成的图片内容如下:

汉字会变成“口口口”的字样,原因是当前系统缺少:STSong-Light 这个字体,我用window系统在本地调试,加了这个字段还是不生效,在测试环境上(linux操作系统)添加这个字体后,显示正常,字体包获取地址:

链接:https://pan.baidu.com/s/1uRxqU9LnaWykd4SumDpC7g 
提取码:te6d 
 

用心编写程序,用智慧照亮人生,fight!fight!fight!

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
Java 通过 pdfbox 库可以实现图片 PDF 的功能,同时也可以通过该库对 PDF 中的图片进行压缩。以下是一个示例代码: ```java import org.apache.pdfbox.pdmodel.PDDocument; import org.apache.pdfbox.pdmodel.PDPage; import org.apache.pdfbox.pdmodel.graphics.image.JPEGFactory; import org.apache.pdfbox.pdmodel.graphics.image.PDImageXObject; import java.io.File; import java.io.IOException; public class ImageToPdf { public static void main(String[] args) throws IOException { // 创建一个空白的PDF文档 PDDocument document = new PDDocument(); // 加载要转换PDF图片文件 File imageFile = new File("image.jpg"); // 创建一个页面对象 PDPage page = new PDPage(); // 将图片转换PDF图像对象 PDImageXObject image = JPEGFactory.createFromImage(document, ImageIO.read(imageFile), 0.5f); // 在页面上添加图像 page.getCropBox(); page.setMediaBox(image.getCropBox()); page.setRotation(0); page.setArtBox(image.getCropBox()); page.setBleedBox(image.getCropBox()); page.setTrimBox(image.getCropBox()); PDPageContentStream contentStream = new PDPageContentStream(document, page); contentStream.drawImage(image, 0, 0); // 关闭页面内容流 contentStream.close(); // 将页面添加到文档中 document.addPage(page); // 保存PDF文件 document.save("image.pdf"); // 关闭PDF文档 document.close(); } } ``` 如果需要在压缩 PDF 中的图片,可以使用下面的代码对 PDF 中的所有图片进行压缩: ```java import org.apache.pdfbox.cos.COSName; import org.apache.pdfbox.pdmodel.PDDocument; import org.apache.pdfbox.pdmodel.PDPage; import org.apache.pdfbox.pdmodel.graphics.image.PDImageXObject; import org.apache.pdfbox.pdmodel.graphics.image.JPEGFactory; import org.apache.pdfbox.pdmodel.graphics.image.LosslessFactory; import org.apache.pdfbox.pdmodel.graphics.image.PDImageXObject; import org.apache.pdfbox.pdmodel.graphics.image.PDImageXObject; import org.apache.pdfbox.pdmodel.graphics.image.PDImageXObject; import java.io.File; import java.io.IOException; public class CompressPdfImages { public static void main(String[] args) throws IOException { // 加载PDF文件 PDDocument document = PDDocument.load(new File("input.pdf")); // 遍历所有页面 for (PDPage page : document.getPages()) { // 获取页面中的所有图像 for (PDImageXObject image : page.getResources().getXObjectNames().stream() .map(x -> page.getResources().getXObject((COSName) x)) .filter(x -> x instanceof PDImageXObject) .map(x -> (PDImageXObject) x) .toArray(PDImageXObject[]::new)) { // 如果图像是JPEG格式,则进行压缩 if ("jpg".equals(image.getSuffix())) { PDImageXObject compressedImage = JPEGFactory.createFromImage(document, image.getImage(), 0.5f); page.getResources().add(compressedImage); page.getResources().remove(image.getName()); } // 如果图像是PNG格式,则进行压缩 else if ("png".equals(image.getSuffix())) { PDImageXObject compressedImage = LosslessFactory.createFromImage(document, image.getImage(), 0.5f); page.getResources().add(compressedImage); page.getResources().remove(image.getName()); } } } // 保存压缩后的PDF文件 document.save("output.pdf"); // 关闭PDF文档 document.close(); } } ``` 如果需要将多个 PDF 文件合并为一个 PDF 文件,可以使用下面的代码: ```java import org.apache.pdfbox.multipdf.PDFMergerUtility; import java.io.File; import java.io.IOException; public class MergePdfFiles { public static void main(String[] args) throws IOException { // 创建一个PDF合并工具对象 PDFMergerUtility merger = new PDFMergerUtility(); // 添加要合并的PDF文件 merger.addSource(new File("file1.pdf")); merger.addSource(new File("file2.pdf")); merger.addSource(new File("file3.pdf")); // 合并PDF文件 merger.setDestinationFileName("merged.pdf"); merger.mergeDocuments(null); } } ``` 以上是 Java 中使用 pdfbox 库实现图片 PDF、压缩 PDF图片、合并多个 PDF 文件为一个 PDF 文件的示例代码。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值