上篇文章我们借助POI导出了EXCEL格式的文件,这次我们换另一种格式PDF,那么,用什么开源组件呢?呵呵,听说iText不错,我们就用iText吧。
首先我们来了解下iText: 简单地说,iText是一个能够快速产生PDF文件的java类库。iText的java类对于那些要产生包含文本,表格,图形的只读文档是很有用的。它的类库尤其与java Servlet有很好的给合。使用iText与PDF能够使你正确的控制Servlet的输出。恩?怎么像教科书啊?呵呵,这就是官网的介绍了。那么,唐伯虎点蚊香的武状元出场了,请问,到哪里可以下载得到呢?下面是该项目主页:http://www.lowagie.com/iText/ 我这里使用的是 2.0.7 版本。
我们已经接触过一个第三方开源组件POI,那么应该积累了一点使用经验了。我们还是利用上篇文章http://blog.csdn.net/lenotang/archive/2008/08/24/2823230.aspx 上的两个业务类:Student.java 和Book.java,有心理准备了吗?呵呵,出代码:
package org.leno.export.util;
import java.io.UnsupportedEncodingException;
public class StrHelp {
public static String getChinese(String s) {
try {
return new String(s.getBytes("gb2312"), "iso-8859-1");
} catch (UnsupportedEncodingException e) {
return s;
}
}
}
package org.leno.export.util;
import java.io.IOException;
import com.lowagie.text.*;
import com.lowagie.text.pdf.BaseFont;
public class PdfParagraph extends Paragraph {
private static final long serialVersionUID = -244970043180837974L;
public PdfParagraph(String content) {
super(content, getChineseFont(12, false));
}
public PdfParagraph(String content, int fontSize, boolean isBold) {
super(content, getChineseFont(fontSize, isBold));
}
// 设置字体-返回中文字体
protected static Font getChineseFont(int nfontsize, boolean isBold) {
BaseFont bfChinese;
Font fontChinese = null;
try {
bfChinese = BaseFont.createFont("c://windows//fonts//simsun.ttc,1",
BaseFont.IDENTITY_H, BaseFont.EMBEDDED);
if (isBold) {
fontChinese = new Font(bfChinese, nfontsize, Font.BOLD);
} else {
fontChinese = new Font(bfChinese, nfontsize, Font.NORMAL);
}
} catch (DocumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return fontChinese;
}
// 转化中文
protected Cell ChangeCell(String str, int nfontsize, boolean isBold)
throws IOException, BadElementException, DocumentException {
Phrase ph = ChangeChinese(str, nfontsize, isBold);
Cell cell = new Cell(ph);
// cell.setBorderWidth(3);
return cell;
}
// 转化中文
protected Chunk ChangeChunk(String str, int nfontsize, boolean isBold)
throws IOException, BadElementException, DocumentException {
Font FontChinese = getChineseFont(nfontsize, isBold);
Chunk chunk = new Chunk(str, FontChinese);
return chunk;
}
// 转化中文
protected Phrase ChangeChinese(String str, int nfontsize, boolean isBold)
throws IOException, BadElementException, DocumentException {
Font FontChinese = getChineseFont(nfontsize, isBold);
Phrase ph = new Phrase(str, FontChinese);
return ph;
}
}
以上是两个帮助封装类,都是为了对付iText的中文问题的。下面就是主角出场:
package org.leno.export.util;
import java.io.*;
import java.lang.reflect.*;
import java.util.*;
import com.lowagie.text.*;
import com.lowagie.text.pdf.*;
import java.awt.Color;
import javax.swing.JOptionPane;
import java.net.MalformedURLException;
import java.text.SimpleDateFormat;
/**
* 利用开源组件IText 2.0.7 动态导出PDF文档 转载时请保留以下信息,注明出处!
*
* @author leno
* @version v1.0
* @param <T>
* 应用泛型,代表任意一个符合javabean风格的类
* 注意这里为了简单起见,boolean型的属性xxx的get器方式为getXxx(),而不是isXxx()
* byte[]表图片数据,注意合适的大小
*/
public class ExportPdf<T> {
public void exportPdf(Collection<T> dataset, OutputStream out) {
exportPdf("测试iText导出PDF文档", null, dataset, out, "yyyy-MM-dd");
}
public void exportPdf(String[] headers, Collection<T> dataset,
OutputStream out) {
exportPdf("测试iText导出PDF文档", headers, dataset, out, "yyyy-MM-dd");
}
public void exportPdf(String[] headers, Collection<T> dataset,
OutputStream out, String pattern) {
exportPdf("测试iText导出PDF文档", headers, dataset, out, pattern);
}
/**
* 这是一个通用的方法,利用了JAVA的反射机制,可以将放置在JAVA集合中并且符号一定条件的数据以PDF 的形式输出到指定IO设备上
*
* @param title
* 表格标题名
* @param headers
* 表格属性列名数组
* @param dataset
* 需要显示的数据集合,集合中一定要放置符合javabean风格的类的对象。此方法支持的
* javabean属性的数据类型有基本数据类型及String,Date,byte[](图片数据)
* @param out
* 与输出设备关联的流对象,可以将PDF文档导出到本地文件或者网络中
* @param pattern
* 如果有时间数据,设定输出格式。默认为"yyy-MM-dd"
*/
@SuppressWarnings("unchecked")
public void exportPdf(String title, String[] headers,
Collection<T> dataset, OutputStream out, String pattern) {
// 作为报表的PDF文件,一定要适合打印机的输出打印
Rectangle rectPageSize = new Rectangle(PageSize.A4);// 定义A4页面大小
// rectPageSize = rectPageSize.rotate();// 加上这句可以实现A4页面的横置
Document document = new Document(rectPageSize, 50, 50, 50, 50);// 其余4个参数,设置了页面的4个边距
try {
// 将PDF文档写出到out所关联IO设备上的书写对象
PdfWriter.getInstance(document, out);
// 添加文档元数据信息
document.addTitle(StrHelp.getChinese(title));
document.addSubject("export information");
document.addAuthor("leno");
document.addCreator("leno");
document.addKeywords("pdf itext");
// 定义页头和页尾
HeaderFooter header = new HeaderFooter(new PdfParagraph(title, 20,
true), false);
header.setAlignment(Element.ALIGN_CENTER);
HeaderFooter footer = new HeaderFooter(new Phrase(
"This is page "), new Phrase("."));
footer.setAlignment(Element.ALIGN_CENTER);
document.setHeader(header);
document.setFooter(footer);
// 打开PDF文档
document.open();
// 添加一张表格,使用Table或者PdfPTable
// Table table = new Table(headers.length);
// table.setWidth(16*headers.length);
// //table.setWidths(new float[]{20,20,20,30});
// table.setCellsFitPage(true);
// table.setAutoFillEmptyCells(true);
// table.setAlignment(Table.ALIGN_CENTER);
// table.setBackgroundColor(Color.yellow);
// table.setBorderColor(Color.green);
PdfPTable table = new PdfPTable(headers.length);
// table.setHorizontalAlignment(Element.ALIGN_CENTER);
table.setWidthPercentage(16 * headers.length);
// 产生表格标题行
for (int i = 0; i < headers.length; i++) {
PdfPCell cell = new PdfPCell(new PdfParagraph(headers[i], 14,
true));
cell.setHorizontalAlignment(Cell.ALIGN_CENTER);
cell.setVerticalAlignment(Cell.ALIGN_MIDDLE);
cell.setBackgroundColor(Color.cyan);
cell.setBorderColor(Color.green);
table.addCell(cell);
}
// 遍历集合数据,产生数据行
Iterator<T> it = dataset.iterator();
int index = 0;
while (it.hasNext()) {
index++;
T t = (T) it.next();
// 利用反射,根据javabean属性的先后顺序,动态调用getXxx()方法得到属性值
Field[] fields = t.getClass().getDeclaredFields();
for (short i = 0; i < fields.length; i++) {
PdfPCell cell = null;
Field field = fields[i];
String fieldName = field.getName();
String getMethodName = "get"
+ fieldName.substring(0, 1).toUpperCase()
+ fieldName.substring(1);
try {
Class tCls = t.getClass();
Method getMethod = tCls.getMethod(getMethodName,
new Class[] {});
Object value = getMethod.invoke(t, new Object[] {});
// 判断值的类型后进行强制类型转换
String textValue = null;
if (value instanceof Boolean) {
boolean bValue = (Boolean) value;
textValue = "男";
if (!bValue) {
textValue = "女";
}
} else if (value instanceof Date) {
Date date = (Date) value;
SimpleDateFormat sdf = new SimpleDateFormat(pattern);
textValue = sdf.format(date);
} else if (value instanceof byte[]) {
byte[] bsValue = (byte[]) value;
Image img = Image.getInstance(bsValue);
cell = new PdfPCell(img);
} else {
textValue = value.toString();
}
// 如果不是图片数据,就当做文本处理
if (textValue != null) {
cell = new PdfPCell(new PdfParagraph(textValue));
}
cell.setHorizontalAlignment(Cell.ALIGN_CENTER);
cell.setVerticalAlignment(Cell.ALIGN_MIDDLE);
cell.setBorderColor(Color.green);
table.addCell(cell);
} catch (SecurityException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (NoSuchMethodException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalArgumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (InvocationTargetException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 清理资源
}
}
}
document.add(table);
document.close();
} catch (DocumentException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public static void main(String[] args) {
// 测试学生
ExportPdf<Student> ex = new ExportPdf<Student>();
String[] headers = { "学号", "姓名", "年龄", "性别", "出生日期" };
java.util.List<Student> dataset = new ArrayList<Student>();
dataset.add(new Student(10000001, "张三", 20, true, new Date()));
dataset.add(new Student(20000002, "李四", 24, false, new Date()));
dataset.add(new Student(30000003, "王五", 22, true, new Date()));
// 测试图书
ExportPdf<Book> ex2 = new ExportPdf<Book>();
String[] headers2 = { "图书编号", "图书名称", "图书作者", "图书价格", "图书ISBN",
"图书出版社", "封面图片" };
java.util.List<Book> dataset2 = new ArrayList<Book>();
try {
BufferedInputStream bis = new BufferedInputStream(
new FileInputStream("book.jpg"));
byte[] buf = new byte[bis.available()];
while ((bis.read(buf)) != -1) {
//
}
dataset2.add(new Book(1, "jsp", "leno", 300.33f , "1234567",
"清华出版社", buf));
dataset2.add(new Book(2, "java编程思想", "brucl", 300.33f , "1234567",
"阳光出版社", buf));
dataset2.add(new Book(3, "DOM艺术", "lenotang", 300.33f , "1234567",
"清华出版社", buf));
dataset2.add(new Book(4, "c++经典", "leno", 400.33f , "1234567",
"清华出版社", buf));
dataset2.add(new Book(5, "c#入门", "leno", 300.33f , "1234567",
"汤春秀出版社", buf));
OutputStream out = new FileOutputStream("E://a.pdf");
OutputStream out2 = new FileOutputStream("E://b.pdf");
ex.exportPdf(headers, dataset, out);
ex2.exportPdf(headers2, dataset2, out2);
out.close();
out2.close();
JOptionPane.showMessageDialog(null, "pdf导出成功!");
System.out.println("pdf导出成功!");
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
恩,有了上一次的经验和教训,相信这次大家看这些代码已经不觉得头疼了。大同小异嘛,连测试和方法的声明都一样。大家结合文档,应该很快就能掌握iText的用法了。那么在web环境下使用该类上篇文章已经介绍过了,我这里就不赘述啦。这样当下次你的项目中要用到数据的导出功能时,你就可以学以致用。
好啦,两个开源组件学习完毕,我这里只是把它们在实际项目中使用的方式列出来了,并没有一点一点介绍它们的API,关于它们系统的学习,还请参考其他的文章和帮助文档。那么,我这里要着重强调一点:我们按照一定的设计思想写出代码,重要的的就不是代码,而是我们的思维!大家要尽可能让自己的代码变得通用灵活并且结构清晰,要学会与代码对话,在项目中更是要侧重设计和功能,而不是技术。始终记住,技术是为业务需求服务的!呵呵,唠叨就到这里了。希望您有所进步!