POI导出Excel工具类

2014-09-30· 5069 次浏览
## 介绍 利用反射机制,在POI的基础上进行了封装,直接传Javabean集合即可导出Excel,可用来做web报表等。可以动态设置列头,列宽,日期格式,可以存储图片。 ## 效果图 ![无标题.jpg](https://image.xsoftlab.net/baike/articleImages/333ff9e76f9340056978fa9c15cf8013.jpg) ## 准备工作 [poi-3.10.1.jar](http://central.maven.org/maven2/org/apache/poi/poi/3.10.1/poi-3.10.1.jar) 与 [commons-codec-1.2.jar](http://central.maven.org/maven2/commons-codec/commons-codec/1.2/commons-codec-1.2.jar) ### maven ```html <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>3.10.1</version> </dependency> <dependency> <groupId>commons-codec</groupId> <artifactId>commons-codec</artifactId> <version>1.2</version> </dependency> ``` ## 代码 ```java package net.xsoftlab.baike; import java.io.BufferedInputStream; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.OutputStream; import java.lang.reflect.Field; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Collection; import java.util.Date; import java.util.Iterator; import java.util.List; import java.util.regex.Matcher; import java.util.regex.Pattern; import org.apache.poi.hssf.usermodel.HSSFCell; import org.apache.poi.hssf.usermodel.HSSFCellStyle; import org.apache.poi.hssf.usermodel.HSSFClientAnchor; import org.apache.poi.hssf.usermodel.HSSFFont; import org.apache.poi.hssf.usermodel.HSSFPatriarch; import org.apache.poi.hssf.usermodel.HSSFRichTextString; import org.apache.poi.hssf.usermodel.HSSFRow; import org.apache.poi.hssf.usermodel.HSSFSheet; import org.apache.poi.hssf.usermodel.HSSFWorkbook; import org.apache.poi.hssf.util.HSSFColor; /** * Excel 工具类(POI) * * @author baike.xsoftlab.net * * @param <T> * 需要存储的类 - 支持基本数据类型及String,Date,byte[](图片数据) * */ public class ExcelUtil<T> { // 表格标题 private String sheetTitle = "Sheet1"; // 列标题及列宽 private String[][] headers = {}; // 数据集 private Collection<T> datasets; // 日期输出格式 private String dateFormat = "yyyy-MM-dd"; // 输出流 private OutputStream out = null; // 图片行行高 public static int PICLINEHEIGHT = 60; public ExcelUtil() { super(); } public ExcelUtil(Collection<T> datasets, OutputStream out) { super(); this.datasets = datasets; this.out = out; } public ExcelUtil(String sheetTitle, Collection<T> datasets, OutputStream out) { this(datasets, out); this.sheetTitle = sheetTitle; } public ExcelUtil(String[][] headers, Collection<T> datasets, OutputStream out) { this(datasets, out); this.headers = headers; } public ExcelUtil(String sheetTitle, String[][] headers, Collection<T> datasets, OutputStream out) { this(sheetTitle, datasets, out); this.headers = headers; } public ExcelUtil(String sheetTitle, String[][] headers, Collection<T> datasets, String dateFormat, OutputStream out) { this(sheetTitle, headers, datasets, out); this.dateFormat = dateFormat; } /** * 利用JAVA的反射机制,将集合中的数据输出到指定IO流中 * * 如有图片,需将图片字段(byte)的顺序与表格中的图片列顺序对应 * * @throws Exception * 异常 */ public void ExportExcel() throws Exception { // 声明一个工作薄 HSSFWorkbook workbook = new HSSFWorkbook(); // 生成一个表格 HSSFSheet sheet = workbook.createSheet(sheetTitle); // 声明一个画图的顶级管理器 HSSFPatriarch patriarch = sheet.createDrawingPatriarch(); // 标题样式 HSSFCellStyle titleStyle = workbook.createCellStyle(); // 设置水平居中 titleStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 设置垂直居中 titleStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER); // 标题字体 HSSFFont titleFont = workbook.createFont(); titleFont.setFontName("微软雅黑"); titleFont.setColor(HSSFColor.BLACK.index); titleFont.setFontHeightInPoints((short) 12); // 把字体应用到当前的样式 titleStyle.setFont(titleFont); // 正文样式 HSSFCellStyle bodyStyle = workbook.createCellStyle(); bodyStyle.cloneStyleFrom(titleStyle); // 正文字体 HSSFFont bodyFont = workbook.createFont(); bodyFont.setFontName("宋体"); bodyFont.setColor(HSSFColor.BLACK.index); bodyFont.setFontHeightInPoints((short) 12); bodyStyle.setFont(bodyFont); int index = 0; HSSFRow row = null; if (headers.length > 0) { // 产生表格标题行 row = sheet.createRow(index++); // 设置行高 row.setHeightInPoints(30f); for (int i = 0; i < headers.length; i++) { HSSFCell cell = row.createCell(i); HSSFRichTextString text = new HSSFRichTextString(headers[i][0]); cell.setCellValue(text); cell.setCellStyle(titleStyle); // 设置列宽 sheet.setColumnWidth(i, Integer.parseInt(headers[i][1]) * 256); } } // 遍历集合数据,产生数据行 Iterator<T> it = datasets.iterator(); while (it.hasNext()) { row = sheet.createRow(index); // 设置行高 row.setHeightInPoints(25f); T t = it.next(); // 利用反射,得到属性值 Field[] fields = t.getClass().getDeclaredFields(); for (int i = 0; i < fields.length; i++) { HSSFCell cell = row.createCell(i); cell.setCellStyle(bodyStyle); Field field = fields[i]; field.setAccessible(true); Object value = field.get(t); // 判断值的类型后进行强制类型转换 String textValue = null; if (value instanceof Date) { Date date = (Date) value; SimpleDateFormat sdf = new SimpleDateFormat(dateFormat); textValue = sdf.format(date); } else if (value instanceof byte[]) { // 设置图片行行高 row.setHeightInPoints(PICLINEHEIGHT); byte[] bsValue = (byte[]) value; HSSFClientAnchor anchor = new HSSFClientAnchor(0, 0, 1023, 255, (short) i, index, (short) i, index); anchor.setAnchorType(2); patriarch.createPicture(anchor, workbook.addPicture(bsValue, HSSFWorkbook.PICTURE_TYPE_JPEG)); } else { // 其它数据类型都当作字符串简单处理 textValue = value.toString(); } // 如果不是图片数据,就利用正则表达式判断textValue是否全部由数字组成 if (textValue != null) { Pattern p = Pattern.compile("^//d+(//.//d+)?$"); Matcher matcher = p.matcher(textValue); if (matcher.matches()) { // 是数字当作double处理 cell.setCellValue(Double.parseDouble(textValue)); } else { cell.setCellValue(textValue); } } } index++; } workbook.write(out); } public static void main(String[] args) throws Exception { // 要导出的图片 BufferedInputStream in = new BufferedInputStream(new FileInputStream("d:/1.jpg")); byte[] buf = new byte[in.available()]; in.read(buf); String[][] headers = { { "序号", "8" }, { "姓名", "20" }, { "年龄", "10" }, { "日期", "15" }, { "头像", "12" } }; List<Student> dataset = new ArrayList<Student>(); dataset.add(new Student(1, "张三", 21, new Date(), buf)); dataset.add(new Student(2, "李四", 22, new Date(), buf)); dataset.add(new Student(3, "王五", 23, new Date(), buf)); // Excel输出地址 OutputStream out = new FileOutputStream(new File("d:/test.xls")); new ExcelUtil<Student>("班级人员", headers, dataset, out).ExportExcel(); System.out.println("excel导出成功!"); in.close(); } } class Student { public long id; public String name; public int age; public Date date; public byte[] b; public Student() { super(); } public Student(long id, String name, int age) { super(); this.id = id; this.name = name; this.age = age; } public Student(long id, String name, int age, Date date) { super(); this.id = id; this.name = name; this.age = age; this.date = date; } public Student(long id, String name, int age, Date date, byte[] b) { super(); this.id = id; this.name = name; this.age = age; this.date = date; this.b = b; } } ``` ## 参考资料 ### 官方API [http://poi.apache.org/apidocs/index.html](http://poi.apache.org/apidocs/index.html) ### 官方开发指南 [http://poi.apache.org/spreadsheet/quick-guide.html](http://poi.apache.org/spreadsheet/quick-guide.html)