問(wèn)題的由來(lái)
前不久做了一個(gè)通過(guò)JSP生成PDF報表的小項目,算得上開(kāi)了一次眼界。企業(yè)的一些信息通過(guò)網(wǎng)絡(luò )形成Html報表,雖然IE可以直接打印顯示在其中的內容,但是從界面上來(lái)看,如果直接將Html的顯示結果打印出來(lái),顯得不太美觀(guān)。如果將它轉成PDF文件再打印,則打印效果會(huì )好很多。
iText簡(jiǎn)介
iText是一個(gè)開(kāi)放源碼的Java類(lèi)庫,可以用來(lái)方便地生成PDF文件。大家通過(guò)訪(fǎng)問(wèn)http://sourceforge.net/project/showfiles.php?group_id=15255&release_id=167948下載最新版本的類(lèi)庫,下載完成之后會(huì )得到一個(gè).jar包,把這個(gè)包加入JDK的classpath即可使用。如果生成的PDF文件中需要出現中文、日文、韓文字符,則還需要通過(guò)訪(fǎng)問(wèn)http://itext.sourceforge.net/downloads/iTextAsian.jar下載iTextAsian.jar包。
關(guān)于iText類(lèi)庫的使用,http://www.lowagie.com/iText/tutorial/index.html有比較詳細的教程。該教程從入門(mén)開(kāi)始,比較系統地介紹了在PDF文件中放入文字、圖片、表格等的方法和技巧。讀完這片教程,大致就可以做一些從簡(jiǎn)單到復雜的PDF文件了。不過(guò),試圖通過(guò)教程解決在生成PDF文件過(guò)程中遇到的所有困難無(wú)疑是一種奢望。所以,閱讀iText的api文檔顯得非常重要。讀者在下載類(lèi)庫的同時(shí),也可以下載類(lèi)庫的文檔。
如何利用iText在java程序中生成PDF報表
以下是上述教程中一個(gè)最簡(jiǎn)單的例子,這個(gè)例子刻畫(huà)了通過(guò)iText生成PDF文件的一般程序框架。讀者只需要在document.open();和document.close();兩條語(yǔ)句中間加入自己希望放在PDF文件中的內容即可。該例子只在PDF文件中加了“Hello World“一行文字。
Document document = new Document();
try
{
PdfWriter.getInstance(document, new FileOutputStream ("Chap0101.pdf"));
document.open();
document.add(new Paragraph("Hello World"));
}
catch(DocumentException de)
{
System.err.println(de.getMessage());
}
catch(IOException ioe)
{
System.err.println(ioe.getMessage());
}
document.close();
由以上的例子可見(jiàn),程序的框架十分清楚明了。然而在PDF中指定文字、圖畫(huà)、表格的位置是一件非常麻煩的事情。除了不斷地在程序中修改位置、然后運行程序、生成PDF文件、觀(guān)察元素在PDF中的位置是否合理這樣的過(guò)程以外,似乎還沒(méi)有其它更好的方法。
如何通過(guò)JSP生成PDF報表
這一部分是在iText的教程中所沒(méi)有的,網(wǎng)上的相關(guān)資料也比較少。我曾在CSDN上看過(guò)有人開(kāi)帖詢(xún)問(wèn)實(shí)現細節,有人回復了實(shí)現的原理:先在服務(wù)器上生成PDF文件,然后用戶(hù)通過(guò)點(diǎn)擊指向PDF文件的超鏈接選擇下載或打開(kāi)。這是一個(gè)思路,或者說(shuō)是思路之一。本文實(shí)現了這個(gè)思路,又給出另外一個(gè)思路并通過(guò)兩種途徑實(shí)現之。
1)直接在服務(wù)器上生成PDF文件。
<%@ page import ="com.lowagie.text.*,com.lowagie.text.pdf.*, java.io.*"%>
<%
String filename = "PDF"+(new Random()).nextInt()+".pdf" ;
Document document = new Document(PageSize.A4);
ServletOutputStream out1 = response.getOutputStream();
try
{
PdfWriter writer = PdfWriter.getInstance(document, new FileOutputStream(filename) );
document.open();
document.add(new Paragraph("Hello World"));
document.close();
}
catch(Exception e){}%>
上面的程序在服務(wù)器上生成了一個(gè)靜態(tài)的PDF文件。顯然,每次運行所得的PDF文件的名稱(chēng)應該是獨一無(wú)二不能有重的。本程序通過(guò)隨機函數來(lái)命名生成的PDF文件。本程序的缺點(diǎn)就是,每次運行都會(huì )在服務(wù)器上產(chǎn)生一個(gè)PDF文件,如果不及時(shí)刪除,數量會(huì )越來(lái)越大,這顯然是站點(diǎn)維護者所不愿意看到的。
2)將PDF文件通過(guò)流的形式輸送到客戶(hù)端的緩存。這樣做的好處是不會(huì )在服務(wù)器上留下任何“遺跡”。
i)直接通過(guò)JSP頁(yè)面生成
<%@
page import="java.io.*,java.awt.Color,com.lowagie.text.*,com.lowagie.text.pdf.*"%>
<%
response.setContentType( "application/pdf" );
Document document = new Document();
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
PdfWriter writer=PdfWriter.getInstance( document, buffer );
document.open();
document.add(new Paragraph("Hello World"));
document.close();
DataOutput output = new DataOutputStream( response.getOutputStream() );
byte[] bytes = buffer.toByteArray();
response.setContentLength(bytes.length);
for( int i = 0; i < bytes.length; i++ )
{
output.writeByte( bytes[i] );
}
%>
ii)通過(guò)Servlet生成
import java.io.*;
import javax.servlet.*;
import javax.servlet.http.*;
import com.lowagie.text.*;
import com.lowagie.text.pdf.*;
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws IOException,ServletException
{
Document document = new Document(PageSize.A4, 36,36,36,36);
ByteArrayOutputStream ba = new ByteArrayOutputStream();
try
{
PdfWriter writer = PdfWriter.getInstance(document, ba);
document.open();
document.add(new Paragraph("Hello World"));
}
catch(DocumentException de)
{
de.printStackTrace();
System.err.println("A Document error:" +de.getMessage());
}
document.close();
response.setContentType("application/pdf");
response.setContentLength(ba.size());
ServletOutputStream out = response.getOutputStream();
ba.writeTo(out);
out.flush();
}
結束
我在項目中采用的是第二種方法。本文的源碼在我的tomcat4上面都是調試通過(guò)的。希望可以給大家帶來(lái)方便。
歡迎大家采用,如需轉載,請注明出處。
聯(lián)系客服