原文地址:点击打开链接

知识点:线程调用,异步线程,异步线程回调函数,C#的类,WSH宿主脚本开发

前言:由于一个客户的项目中需要将WORD文档转换成PDF格式,故写了本篇实站教程 
需求分析:客户的项目以B/S结构为主,提供一个WORD文件在后台自动转换成PDF,经过实际测试,如果该篇WORD文档有100多页的话,转换需要20分钟左右的时间(环境:CPU是奔腾M 1.6G,512M内存),整个CPU的占用率近乎95%~100%,此结果告诉客户以后,客户提议:到客户下班后,自动转换PDF,同时如果使用人确认要查看该PDF文档,如果没有转换,提供给客户选择,是现在转换成PDF,还是由服务器在客户下班后,自动转换.

项目功能:按需求分析要写两个功能, 
第一为:B/S结构后台转换,要提交给客户选择 
第二为:Windows服务自动转换WORD文档到PDF

这两个分类:核心的转换程序都是采用线程的方式执行,只不过第一个功能是针对一个WORD文件,第二个功能针对所有未转换的WORD文档.

分析到现在:我们开始实战转换了!

一:必备工具 
安装必须的工具MS VS.Net2003,MS Office2003,Adobe Acrobat 7.0 Professional,postscript.exe,gs811w32.exe

MS VS.Net2003的安装不说明 
MS Office2003的安装不说明

postscript.exe默认安装就可以了,它是一个PDF转换时所需要的脚本 
gs811w32.exe默认安装就可以,它其实是个PDF虚拟打印机的驱动

(postscript.exe和gs811w32.exe可在本站资源区下载)

二:配置虚拟打印机 
进入WINDOWS的控制面板,进入打印机,点击"添加打印机"图标.在安装对话框上"按一步",出现选择打印机时,在制造商一栏中选择"Generic",在打印机一栏中,选择"MS Publisher Color Printer",然后一路按下一步,直到安装结束.

三:开始写第一个程序(脚本程序) 
为什么要使用脚本程序进行转换呢,其实实际测试过程中,使用PDF Distiller的对象引用到C#后,转换成功,但整个PDF Distiller对象不能释放,第二次再转换时,就发生了错误,故此处使用脚本程序

实现转换.这样我们只要在C#的程序中调用脚本程序就可以实现WORD到PDF的转换

宿主脚本文件名:ConvertDoc2PDF.js 
------------------------------------------------ 
脚本文件内容:

var files = WScript.Arguments; 
var fso = new ActiveXObject("Scripting.FileSystemObject"); 
var word = new ActiveXObject("Word.Application"); 
var PDF = new ActiveXObject("PDFDistiller.PDFDistiller.1"); 
word.ActivePrinter = "MS Publisher Color Printer";

//files(0) 为WORD文档文件名 
//files(1) 为,转换后需要保存的路径 
//调用fso.GetBaseName(files(0))后,为无路径,无扩展名,的文件名 
//files.length为文件参数的个数,使用循环可以支持多个WORD文档的转换

var docfile = files(0); 
var psfile = files(1) + fso.GetBaseName(files(0)) + ".ps"; 
var pdffile = files(1) + fso.GetBaseName(files(0)) + ".pdf"; 
var logfile = files(1) + fso.GetBaseName(files(0)) + ".log";

try{ 
var doc = word.Documents.Open(docfile); 
//WORD文件转成PS文件; 
word.PrintOut(false, false, 0, psfile); 
doc.Close(0);

//PS文件转成PDF文件; 
PDF.FileToPDF(psfile,pdffile,"");

fso.GetFile(psfile).Delete();//删除PS脚本文件 
fso.GetFile(logfile).Delete();//删除转换的日志文件

word.Quit(); 
WScript.Echo("isuccess");//成功 
WScript.Quit(0); 

catch(x) 

word.Quit(); 
WScript.Echo("isfail");//失败 
WScript.Quit(0); 
}

然后测试该脚本程序 
启动MS-DOS,输入如下命令: 
c:\>cscript //nologo c:\ConvertDoc2PDF.js c:\test.doc c:\

说明: 
运行成功后将看到test.pdf文档了 
c:\test.doc参数对应的是脚本程序中的files(0) 
c:\参数对应的是脚本程序中的files(1)

你可以安照该脚本改写成,支持多个参数,使用FOR循环,一次转换多个WORD文档,此处没有使用多个文件转换功能,是考虑到,该段脚本放在C#的线程中执行,这样一来也可以转换多个WORD文档.

四:使用C#调用ConvertDoc2PDF.js脚本 
新建一个C#的WINDOWS应用程序,添加一个按钮button1 
添加一个函数,函数名StartConvertPDF 
public void StartConvertPDF() 

Process proc = new Process(); 
proc.StartInfo.FileName = "cmd.exe"; 
proc.StartInfo.WorkingDirectory = @"c:\"; 
proc.StartInfo.CreateNoWindow = true; 
proc.StartInfo.UseShellExecute = false; 
proc.StartInfo.RedirectStandardInput = true; //输入重定向

proc.Start(); 
proc.StandardInput.WriteLine(@"cscript //nologo c:\ConvertDoc2PDF.js c:\test.doc c:\"); 
proc.StandardInput.WriteLine("exit"); 
proc.WaitForExit(); 
}

然后在按钮的CLICK事件中添加调用线程的代码 
private void button1_Click(object sender, System.EventArgs e) 

//定义线程序 
Thread thConvert = new Thread(new ThreadStart(StartConvertData)); 
thConvert.Start(); 
}

注意:在测试上面的C#程序时,必须添加如下命名空间 
using System.Diagnostics; 
using System.Threading;

五:健壮的C#调用代码(实际考虑,可放在B/S系统中) 
完成第4步的C#测试后,细心的读者,可能看到一点问题,那就是如何得到脚本运行后输出的结果,如何给线程中调用的StartConvertData方法传递参数 
1:传递参数,此话说来也可用一篇教程告诉大家线程中方法如何来传递参数,现在就讲一个方案,此种方案很多,我采用一个类,初始化这个类,然后调用该类的方法作为线程执行的方法 
2:得到脚本的输出结果,使用Process对象的输出重定向,就是说改变输出方向,使脚本不输出到控制台(MS-DOS窗口),而是重定向输出到C#程序中,并采用线程的异步回调方法,显示脚本运行结果

添加一个新类,类名为ToPdf 
using System; 
using System.Diagnostics; 
using System.ComponentModel; 
using System.Windows.Forms; 
using System.Data;

namespace Doc2Pdf 

public class ToPdf 

private string strWord = "";//此处的WORD文件不含路径 
private string sPath = ""; 
public string sExecResult = ""; 
public bool bSuccess = false;

public ToPdf(string sParamWord,string sParamPath) 

strWord = sParamWord; 
sPath = sParamPath; 
}

public void StartConvertPDF() 

Process proc = new Process(); 
   proc.StartInfo.FileName = "cmd.exe"; 
   proc.StartInfo.WorkingDirectory = sPath; 
   proc.StartInfo.CreateNoWindow = true; 
   proc.StartInfo.UseShellExecute = false; 
   proc.StartInfo.RedirectStandardInput = true;//标准输入重定向 
proc.StartInfo.RedirectStandardOutput = true;//标准输出重定向

   proc.Start(); 
proc.StandardInput.WriteLine("cscript //nologo "+sPath+"ConvertDoc2PDF.js "+sPath+strWord+ " "+sPath); 
proc.StandardInput.WriteLine("exit"); 
sExecResult = proc.StandardOutput.ReadToEnd();//返回脚本执行的结果 
proc.WaitForExit(); 
proc.Close();

}

public void EndConvertPDF(System.IAsyncResult ar)//ar参数必须写,是线程执行完成后的回调函数 

if(sExecResult.IndexOf("isuccess")!=-1)bSuccess=true; 
else if(sExecResult.IndexOf("isfail")!=-1)bSuccess=false; 
//如果放在B/S系统,你可以在此处写数据库,是成功还是失败,并用一个WEBService程序不断检查数据库,此WEBService程序不放在该回调用函数中 
//如果放在C/S系统,回调函数可以不放在类中,以便在窗体程序中调用结果 


}

改写原来的button1_Click事件中的代码 
------------------------------------------------------------- 
private void button1_Click(object sender, System.EventArgs e) 

ToPdf my2Pdf = new ToPdf("test.doc","c:\\"); 
ThreadStart thStartConvert = new ThreadStart(my2Pdf.StartConvertPDF); //开始异步调用线程 
thStartConvert.BeginInvoke(new AsyncCallback(my2Pdf.EndConvertPDF),null);//设置异步线程的回调函数

//如果需要转换多个WORD,你可以用循环 
//如果是B/S系统,可以将本段代码放在ASPX中,并结合客户端的无刷新显示数据的技术,不断访问WEBService程序,以确定PDF是否转换成功或失败 
}

六:编写更加健壮的C#调用代码(实际考虑,可放在WINDOWS的服务程序中) 
--------------------------------------------------------------- 
实际使用时,由于转化PDF时CPU的占用率很高,考虑只在同一时间转换一篇WORD文档,放弃异步线程的回调函数的使用,考虑一个WINDOWS的服务程序. 
写一个函数CheckData2Convert(),不断的检查没有转换的WORD文档,并使用循环调用ToPdf类中执行转换方法StartConvertPDF

//以下给出,泛代码,用户按照自己的需求,填写完整即可 
//bool bStart为全局变量,控制循环的进入与退出 
//例:18:30开始检查并转换,那么18:30时,bStart=true;并启动转换线程 
//6:30停止转换线程,bStart=fasle;

private void CheckData2Convert() 

//检查指定目录下的没有转换的WORD文档,你同样可以检查数据库中记录的没有转换的WORD文档 
string sPath = System.Threading.Thread.GetDomain().BaseDirectory; //当前的路径 
while(bStart) 

int iFileCount = CheckWord(); //CheckWord为一个方法,检查当前没有转换的WORD文档,返回没有转换的文件数,该方法的代码由读者自己编写 
for(int i=0;i{ 
string sWord = GetWordFileName(i) //GetWordFileName为一个方法,返回一个不带路径的WORD文件名,该方法的代码由读者自己编写 
//ToPdf类中的StartConvertPDF()方法使用的是不带路径的WORD文件名 
ToPdf my2Pdf = new ToPdf(sWord ,sPath); 
my2Pdf.StartConvertPDF();

if(my2Pdf.sExecResult.IndexOf("isuccess")!=-1) 

//成功,写日志,或回写数据库 

else if(my2Pdf.sExecResult.IndexOf("isfail")!=-1) 

//失败,写日志,或回写数据库 
}

}

if(!bStart)break; 
Thread.Sleep(1000); 

}

然后在服务的开始事件中,启动线程 
protected override void OnStart(string[] args) 

//可以使用一个开始定时器,检查是否到开始时间,时间一到,就开始执行线程,此处的开始执行线程可以放在开始定时事件中 
//可以使用一个结束定时器,检查是否到结束时间,时间一到,就结束线程,结束线程的代码可以放在结束定时事件中 
//注意:应该使用组件中的定时器,而不是Windows的FORMS中的定时器 
//该定时器的类名为System.Timers.Timer,千万别搞错,不然执行不会正常的 
bStart = true; 
Thread thConvert = new Thread(new ThreadStart(StartConvertData)); 
thConvert.Start(); 
}

然后在服务的结束事件中,设置停止线程的标识bStart= false 
protected override void OnStop() 

bStart = false; 
//为何此处不停止线程呢,因为考虑到,现在线程正在转换WORD文档,但没有结束,所以只设置停止标识,转换完成后,线程也执行结束了. 
}

实战用C#将Word转换PDF格式相关推荐

  1. Word转换PDF格式的C#或ASP.Net实战

    <script language='javascript' src='http://www.shiqiaotou.com/donetk/Header.js'></script> ...

  2. ASP.Net实现将Word转换PDF格式

    前言:由于一个客户的项目中需要将 Word文档 转换成 PDF 格式,故写了本篇实站教程 需求分析:客户的项目以B/S结构为主,提供一个Word文件在后台自动转换成PDF,经过实际测试,如果该篇Wor ...

  3. Word转换PDF格式的C#第二版(不再使用WSH宿主脚本)

    第一个版本,查看 为什么要写第二个版本呢,当时再完成项目时,比较赶,没有多考虑,就借用cscript方式写了个脚本,现在该项目已经开完评审会,有时间重新将该程序重写,直接使用纯C#来完成. 当时由于在 ...

  4. java : word,excel,img,ppt各种文档转换pdf格式以流方式

    前提: 面对各种文件转换pdf格式,我下面写的都是一些方法,其中每个方法都以流的方式进行参数的传递. 通过猿友的帮忙,修改了部分jar包,解决excel转换pdf导致的水印问题~ 源码链接:https ...

  5. Linux系统Word转换PDF,文档字体乱码不显示问题解决。

    1.问题 在Windows系统中,使用Java将Word文档转换为PDF格式时,结果文档内容正常:但是在Linux系统中,转出来PDF文档出现乱码或###符号等. 2.问题原因 这个问题是由于Linu ...

  6. 关于poi操作word,word转换pdf预览,这边文章就够了

    最近公司一个项目中需要导出一个简历word版,对于实习生的我开始是无从下手的,后来通过搜索发现可以使用poi来操作. 话不多说,先引入依赖 <dependency><groupId& ...

  7. Java使用Spire.Pdf或Aspose-Words实现Word转换Pdf在Linux服务器上的中文乱码问题

    一.问题产生的背景 当Word文档中含有中文字符时,不管是使用Spire.Pdf或Aspose-Words实现Word转换Pdf,代码的执行都需要调用操作系统的本地字体库支持,否则所有中文字符都将乱码 ...

  8. ppt转换pdf格式软件

    为什么80%的码农都做不了架构师?>>> ppt转换pdf格式软件 最近小编就曾遇到了这么一个情况,老板搞到一份100多页的ppt格式的文件,由于删除.编写的不方便,让我将这篇文档转 ...

  9. 报名系统网页导出html,【网页报名表如何导出pdf】_网页的报名表怎样转换为word或者PDF格式...

    网页的报名表怎样转换为word或者pdf格式 网页的报名表怎样转换为word或者pdf格式呢?小编将为大家带来以下三个方法,希望能对大家有所帮助. 方法一. 1.保存网页 在浏览器来中访问目标网页,执 ...

最新文章

  1. HTTP POST慢速DOS攻击初探
  2. HTML 中有用的字符实体
  3. 切换Python2和Python3的方法
  4. 论文写作常见错误(1)
  5. Check task status after 2016 Spring festival
  6. pandas:根据行间差值进行数据合并
  7. 写flash,要不要加个判断?
  8. 《objective-c基础教程》学习笔记(四)—— OC面向对象编程初探
  9. SpringBoot整合Shiro实现权限控制,验证码
  10. DP-最大递增子序列与最大递增子数组; 最大公共子序列与最大公共子数组。
  11. World Leading!世界第二!百度云在吹牛?
  12. GoFrame v1.6.0 发布,Go 应用开发框架
  13. utilities(matlab)—— 合成数据(synthesis data)
  14. android 权限库EasyPermissions
  15. 知乎日报富文本处理android,一个追求高效的学习者手机里装有哪些APP?
  16. python中encode用法_Python中encode()方法的使用简介
  17. matlab图像自动标注程序,基于MATLAB/GUI的图像语义自动标注系统
  18. 解决打开word时显示microsoft visual basic运行时错误没有注册类的问题
  19. Android Studio导入第三方sdk
  20. 一步步提高手写数字的识别率(1)

热门文章

  1. 学考计算机模拟题,自学考试:计算机原理模拟试题
  2. oracle数据库创建库并使用库,docker, oracle创建空间表,创建数据库。
  3. [2021.10.28]<呆头熊的开发日记>怪物AI
  4. [AHK]文华财经 补数据 自动点击 下载按钮
  5. 郑州万向置业oa服务器信息,[办公OA]-致远OA控件安装说明
  6. ideapad320_朋友买电脑被套路,联想Ideapad320的配置卖的比320S还贵近千元
  7. python如何下载zip文件_Python 使用requests下载zip文件
  8. 基于单片机的智能导盲(拐杖)系统设计
  9. ubuntu如何调整界面大小
  10. Karateclub包介绍及节点嵌入实现