在web项目中使用jacob的问题
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在web项目中使用jacob的问题相关的知识,希望对你有一定的参考价值。
在web项目下,用jacob实现将word文档转换为html格式的页面,方法:
/**
* 文档转换函数
*
* @param docfile
* word文档的绝对路径加文件名(包含扩展名)
* @param htmlfile
* 转换后的html文件绝对路径和文件名(不含扩展名)
*/
public static void change(String docfile, String htmlfile)
ActiveXComponent app = new ActiveXComponent("Word.Application"); // 启动word
try
app.setProperty("Visible", new Variant(false));
// 设置word不可见
Object docs = app.getProperty("Documents").toDispatch();
Object doc = Dispatch.invoke((Dispatch) docs, "Open", Dispatch.Method,
new Object[] docfile, new Variant(false),
new Variant(true) , new int[1]).toDispatch();
// 打开word文件
Dispatch.invoke((Dispatch) doc, "SaveAs", Dispatch.Method, new Object[]
htmlfile, new Variant(8) , new int[1]);
// 作为html格式保存到临时文件
Variant f = new Variant(false);
Dispatch.call((Dispatch) doc, "Close", f);
catch (Exception e)
e.printStackTrace();
finally
app.invoke("Quit", new Variant[] );
在用main函数测试时没有发现问题,但在实际调用时报错:
C:\WINDOWS\system32\jacob.dll already loaded in another classloader
或
com.jacob.com.ComFailException: Invoke of: Open
Source: Microsoft Word
Description:
这是为什么呢
String docfile= "C:\\Program Files\\apache-tomcat-6.0.30\\webapps\\upload4\\word\\"; 参考技术A com.jacob.com.ComFailException: Invoke of: Open
Source: Microsoft Word
Description:
是路径错误 参考技术B 提问者已解决了吗?同求,急求!
使用Jacob将Excel转换PDF问题总结
hello,大家好,我是灰小猿,一个超会写bug的程序猿!
好久不见,分享一个近期在项目开发中遇到的一个新问题,关于使用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);
//设置缩放,值为100或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();
问题二、大量数据时,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时遇到的两个问题,暂且做这两个记录,如果小伙伴们还有其他问题,可以一起交流!
我是灰小猿,我们下期见!
以上是关于在web项目中使用jacob的问题的主要内容,如果未能解决你的问题,请参考以下文章