jacob简介
jacob操作文档转换底层还是调用windows的 office去转换,office是建立在windows平台之上的,本身是一个软件,除了他自己提供的宏似乎没有什么能对他进行直接的操作,在windows平台上为了解决像这样的不同应用软件,通信缺乏通用api问题,推出了com的解决方案。我们使用dll中的一组或多组相关的函数存取组件数据,总的合称为接口(对应jacob,就是Dispatch),具体到每个细节的实现称为方法。我们使用jacob就是通过一个接口来操作word的ActiveX对象(实质是调用指向接口的指针,这也是唯一途径)。
所以在使用jacob进行文档转换时需要用到jacob-1.19-x64.dll;jacob-1.19-x86.dll去做底层的操作。
核心类介绍
- JacobObject:用于Java程序MS下的COM进行通信,创建标准的API框架
- ComThread:初始化COM组件线程,释放线程,对线程进行管理
- Dispatch:调度处理类,封装了操作来从而操作Office,并表示不同MS级别调度对象
- ActiveXComponent : 创建COM组件
- Variant : 与COM通讯的参数或者返回值
- ROT :Running Object Table (ROT),运行对象表将每个线程映射到所有jacobobjects,在线程创建核心方法
核心方法介绍
- Dispatch : 可调用该自动化对象的属性或方法,具体的属性和方法要看参考文档VBA API
- Dispatch.get(dispatch, String name);获取对象属性
- Dispatch.put(dispatch, String name, Object value);设置对象属性
- Dispatch.call(dispatch, String name, Object… args);调用对象方法
类图介绍
使用流程
使用步骤
- 下载jar包https://pan.baidu.com/s/1cilYcypa8Vt-5-uNIqXDIg 提取码:16fl
- 下载dll文件,放在jdk 的bin目录下。(https://pan.baidu.com/s/1PumFrMC9fahEzmvdnz8gRQ 提取码:zsh2)
- 本机电脑中要有 office全家桶 word、xls,ppt 什么的.....
- 注意jdk版本什么的,以免出问题。
试例代码
//word转pdf
public static void doc2PDF(String srcFilePath, String pdfFilePath) throws Exception {
ActiveXComponent app = null;
Dispatch doc = null;
try {
//初始化COM组件线程
ComThread.InitSTA();
//创建com组件,打开word应用程序
app = new ActiveXComponent("Word.Application");
//设置word不可见
app.setProperty("Visible", new Variant(false));
//获得word中所有打开的文档,返回Documents对象
Dispatch docs = app.getProperty("Documents").toDispatch();
Object[] obj = new Object[] { srcFilePath, new Variant(false), new Variant(false), // 是否只读
new Variant(false), new Variant("pwd") };
doc = Dispatch.invoke(docs, "Open", Dispatch.Method, obj, new int[1]).toDispatch();//
//Dispatch.put(doc, "Compatibility", false); //兼容性检查,为特定值false不正确
Dispatch.put(doc, "RemovePersonalInformation", false);
// word保存为pdf格式宏,值为17
Dispatch.call(doc, "ExportAsFixedFormat", pdfFilePath, WORD_TO_PDF_OPERAND);
} catch (Exception e) {
e.printStackTrace();
throw e;
} finally {
if (doc != null) {
Dispatch.call(doc, "Close", false);
}
if (app != null) {
app.invoke("Quit", new Variant[] {});
}
//结束后关闭线程组件
ComThread.Release();
}
}
//ppt转pdf
public static void ppt2PDF(String srcFilePath, String pdfFilePath) throws Exception {
ActiveXComponent app = null;
Dispatch ppt = null;
try {
ComThread.InitSTA();
app = new ActiveXComponent("PowerPoint.Application");
Dispatch ppts = app.getProperty("Presentations").toDispatch();
/*
* call param 4: ReadOnly param 5: Untitled指定文件是否有标题 param 6: WithWindow指定文件是否可见
*/
ppt = Dispatch.call(ppts, "Open", srcFilePath, true, true, false).toDispatch();
Dispatch.call(ppt, "SaveAs", pdfFilePath, PPT_TO_PDF_OPERAND); // ppSaveAsPDF为特定值32
} catch (Exception e) {
e.printStackTrace();
throw e;
} finally {
if (ppt != null) {
Dispatch.call(ppt, "Close");
}
if (app != null) {
app.invoke("Quit", new Variant[] {});
}
ComThread.Release();
}
}
//excel转pdf
public static void excel2PDF(String inFilePath, String outFilePath) throws Exception {
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 = new Object[] { inFilePath, new Variant(false), new Variant(false) };
excel = Dispatch.invoke(excels, "Open", Dispatch.Method, obj, new int[9]).toDispatch();
// 转换格式
Object[] obj2 = new Object[] { new Variant(EXCEL_TO_PDF_OPERAND), // PDF格式=0
outFilePath, new Variant(0) // 0=标准 (生成的PDF图片不会变模糊) ; 1=最小文件
};
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();
}
}
注:路径问题
inFilePath:当前文档所在位置;outFilePath:转pdf后的文件保存位置
如部署在tomcat的某个文件下:
ServletContext servletContext = getRequest().getSession().getServletContext();
String realPath = servletContext.getRealPath("/");
String inFilePath = realPath + "/data/supp/file_upload/PRJFILE/"+recordId;
String outFilePath = realPath +"/data/supp/file_upload/PRJFILE/pdf/"+recordId+".pdf";