利用Jacob将Excel转换PDF的问题汇总

网友投稿 357 2022-07-27


目录前言问题一、Excel数据列较多时,PDF中列打印不全,问题二、大量数据时,PDF页模糊解决总结

前言

好久不见,分享一个近期在项目开发中遇到的一个新问题,关于使用easyexcel生成Excel,并且使用jacob转换成PDF的需求,最开始的时候在网上找了一些相关的教程,经过筛选之后发现还是使用jacob调用office软件来进行转换是最可靠的。然后就和大家出了一篇关于使用jacob将Excel文件转换PDF的教程,Excel转换PDF两种方法总结

但是后来我在实践操作中发现,原来的代码只能实现基本转换,对于一些较为复杂或数据量较大的代码就会出现一些问题,

使用Jacob转换的基本操作和网上其他的教程基本类似,在这里我主要记录一下我在转换中遇到的一些问题,以及在最基本的Excel转PDF的代码的基础上增加的一些东西。

问题一、Excel数据列较多时,PDF中列打印不全,

对于一些比较复杂的数据表,数据列非常的多,有时候按照正常设置在转换PDF的时候,经常会出现列显示不全,部分列溢出的情况。 针对这个问题,可以在Excel转换PDF中做一些设置。其中设置项及其参数可以在如下的代码中设置:

Dispatch.put(value, "key",);

在这里我设置了缩放、纵向打印、行列等属性来解决上面的问题,这些设置可以在下面的代码中看到,大家可以直接在自己的转换代码对应位置做同样的设置。当然我下面的代码是直接配置好的,可以直接复制使用。

转换功能:

package com.gyg.util;

import com.jacob.activeX.ActiveXComponent;

import com.jacob.com.ComThread;

import com.jacob.com.Dispatch;

import com.jacob.com.Variant;

import com.spire.xls.FileFormat;

import com.spire.xls.Workbook;

import com.spire.xls.Worksheet;

import lombok.extern.slf4j.Slf4j;

/**

* @author YunGang.Guo

* @date 2022/01/05 16:44

**/

@Slf4j

public class ExcelToPDFUtil {

/**

* 使用jacob实现excel转PDF

*

* @param inputFilePath 导入Excel文件路径

* @param outputFilePath 导出PDF文件路径

*/

public static void jacobExcelToPDF(String inputFilePath, String outputFilePath) {

ActiveXComponent ax = null;

Dispatch excel = null;

try {

ComThread.InitSTA();

ax = new ActiveXComponent("Excel.Application");

ax.setProperty("Visible", new Variant(false));

//禁用宏

ax.setProperty("AutomationSecurity", new Variant(3));

Dispatch excels = ax.getProperty("Workbooks").toDispatch();

Object[] obj = {

inputFilePath,

new Variant(false),

new Variant(false)

};

excel = Dispatch.invoke(excels, "Open", Dispatch.Method, obj, new int[9]).toDispatch();

//获取到sheets的集合对象

Dispatch sheets = Dispatch.get(excel, "sheets").toDispatch();

//获取到总表数

int count = Dispatch.get(sheets, "count").changeType(Variant.VariantInt).getInt();

for (int i = 1; i <= count; i++) {

//获取到sheet页

Dispatch sheet = Dispatch.invoke(sheets, "Item", Dispatch.Get, new Object[]{i}, new int[1]).toDispatch();

Dispatch page = Dispatch.get(sheet, "PageSetup").toDispatch();

//是否设置区域打印

Dispatch.put(page, "PrintArea", false);

//设置横向打印还是纵向打印

Dispatch.put(page, "Orientation", 2);

//设置缩放,值为1CYKRvep00或false

Dispatch.put(page, "Zoom", false);

//所有行为一页

Dispatch.put(page, "FitToPagesTall", false);

//所有列为一页(1或false)

Dispatch.put(page, "FitToPagesWide", 1);

log.info("sheet页[" + i + "]设置成功,");

}

//转换格式

Object[] obj2 = {

//PDF格式等于0

new Variant(0),

outputFilePath,

//0=标准(生成的PDF图片不会模糊),1=最小的文件

new Variant(0)

};

Dispatch.invoke(excel, "ExportAsFixedFormat", Dispatch.Method, obj2, new int[1]);

} catch (Exception e) {

e.printStackTrace();

throw e;

} finally {

if (excel != null) {

Dispatch.call(excel, "Close", new Variant(false));

}

if (ax != null) {

ax.invoke("Quit", new Variant[]{});

ax = null;

}

ComThread.Release();

}

}

}

问题二、http://大量数据时,PDF页模糊解决

在平常项目中需要导出的Excel表中的数据量是非常大的,对于这种情况,一般在转换的PDF中一张表对应一页是不太可能的。但是Jacob转换时默认就是让一个表在一页上,这样就导致了数据会被缩放的特别小,导致数据模糊。 对于这种情况,我们一般可以在写Excel的处理器中增加一些设置,让Excel在转换PDF的时候,可以自适应PDF页,并且对于一页存放不下的数据,自动分配到下一页。同时设置打印时每一页上都增加标题行。

我在这里是使用了easyexcel生成Excel,并且使用了一个单独的处理器,小伙伴们在使用的时候,也可以将打印PDF的设置项作为一个单独的处理器去使用。该处理器的完整代码如下:

package com.gyg.util;

import com.alibaba.excel.write.handler.SheetWriteHandler;

import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;

import com.alibaba.excel.write.metadata.holder.WriteWorkbookHolder;

import org.apache.poi.ss.usermodel.Sheet;

import org.apache.poi.ss.util.CellRangeAddress;

/**

* @author YunGang.Guo

* @date 2022/04/20 11:30

**/

public class PrintExcelHandler implements SheetWriteHandler {

/**

* 该sheet页的表头列数,用于冻结表头

*/

int handColNum;

PrintExcelHandler(int handColNum){

this.handColNum = handColNum;

}

@Override

public void beforeSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {

}

@Override

public void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {

Sheet sheet = writeSheetHolder.getSheet();

//冻结表头,设置打印的每一页上都加上标题行

CellRangeAddress cellRangeAddress = new CellRangeAddress(0, 0, 0, handColNum);

sheet.setRepeatingRows(cellRangeAddress);

//设置横向打印

sheet.getPrintSetup().setLandscape(true);

sheet.getPrintSetup().setFitHeight((short) 0);

//设置所有列为一页

sheet.setFitToPage(true);

//设置是否显示网格线

sheet.setDisplayGridlines(false);

}

}

以上就是我在使用easyexcel生成Excel,并且使用jacob转换成PDF时遇到的两个问题,暂且做这两个记录,如果小伙伴们还有其他问题,可以一起交流!

总结


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Spring AOP操作的相关术语及环境准备
下一篇:Spring学习通过AspectJ注解方式实现AOP操作(spring aspectj注解)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~