Java POI实现Excel导入导出
POI
简介
Apache POI是Apache软件基金会的开放源码函式库,POI提供API给Java程序对Microsoft Office格式档案读和写的功能。POI为“Poor Obfuscation Implementation”的首字母缩写,意为“简洁版的模糊实现”。
版本依赖
本次使用的poi版本为 4.1.2,部分操作类和之前相比有一些变化。引入了两个jar,其中poi是操作03版及之前excel的(后缀为xls),poi-ooxml是处理07版及之后excel的(后缀为xlsx)。区别就是 xls也就是03版单个sheet的最大行数为65536,xlsx也就是07版之后单个sheet的最大行数为1048576。
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi</artifactId>
<version>4.1.2</version>
</dependency>
<dependency>
<groupId>org.apache.poi</groupId>
<artifactId>poi-ooxml</artifactId>
<version>4.1.2</version>
</dependency>
excel常用类
Excel 03
工作簿: HSSFWorkbook
工作表: HSSFSheet
行 : HSSFRow
单元格: HSSFCell
单元格字体: HSSFFont
工作表头: HSSFHeader
工作表尾: HSSFFooter
单元格样式: HSSFCellStyle
日期工具类: HSSFDateUtil
日期格式: HSSFDataFormat
Excel 07
工作簿: XSSFWorkbook
工作表: XSSFSheet
行 : XSSFRow
单元格: XSSFCell
单元格字体: XSSFFont
工作表头: XSSFHeader
工作表尾: XSSFFooter
单元格样式: XSSFCellStyle
日期工具类: XSSFDateUtil
日期格式: XSSFDataFormat
- 2.1基于07版本的优化,占用内存比较小
工作簿: SXSSFWorkBook
工作表 Sheet
行: Row
单元格: Cell
excel基本操作流程
excel导出03版
//03版
@Test
void poiBase1(){
//1.创建工作簿对象
HSSFWorkbook workbook = new HSSFWorkbook();
//2.创建工作表对象
HSSFSheet sheet1 = workbook.createSheet("sheet1");
//3.创建行,参数为行数,从0开始
HSSFRow row = sheet1.createRow(0);
//单元格类型,数值类型 NUMERIC(整数、小数、日期) 字符串类型String 布尔类型Boolean,
// 默认是空值 Blank createCell(intNum,type);
//创建列(1,1)第一行第一列
// _NONE(-1),
// NUMERIC(0),
// STRING(1),
// FORMULA(2),
// BLANK(3),
// BOOLEAN(4),
// ERROR(5);
HSSFCell cell1 = row.createCell(0,CellType.NUMERIC);
cell1.setCellValue("数据");
HSSFCell cell2 = row.createCell(1);
cell2.setCellValue(new DateTime().toString("yyyy-MM-dd"));
OutputStream os = null;
try {
os = new FileOutputStream(new File("D://test.xls"));
workbook.write(os);
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
assert os != null;
os.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
excel导入03版
/**
* excel导入
*/
@Test
void poiBase3(){
FileInputStream fis;
try {
fis = new FileInputStream(new File("D://test.xls"));
HSSFWorkbook workbook = new HSSFWorkbook(fis);
// HSSFWorkbook workbook1 = new HSSFWorkbook(POIFSFileSystem.create(new File("D://test.xls")));
//获取工作表的个数
int numberOfSheets = workbook.getNumberOfSheets();
for(int i = 0;i<numberOfSheets;i++){
HSSFSheet sheetAt = workbook.getSheetAt(i);
int lastRowNum = sheetAt.getLastRowNum();
List<Object> list = new ArrayList<>();
for(int j =0;j<= lastRowNum;j++){
HSSFRow row = sheetAt.getRow(j);
short lastCellNum = row.getLastCellNum();
Object o[] = new Object[lastCellNum];
for(int k = 0;k<lastCellNum;k++){
o[k] = row.getCell(k);
}
list.add(Arrays.asList(o));
}
System.out.println("第"+i+1+"个sheet内容:\n"+list.toString());
}
} catch (IOException e) {
e.printStackTrace();
}
}
导出excel工具类
package com.pec.util;
import java.io.IOException;
import java.io.OutputStream;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.Collection;
import java.util.Date;
import java.util.Iterator;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.xssf.streaming.SXSSFWorkbook;
import org.apache.poi.xssf.usermodel.*;
public class ExcelExport<T> {
/**
* 利用了JAVA的反射机制,将放置在JAVA集合中并且符合一定条件的数据以EXCEL 的形式输出到指定IO设备上
* @param title 表格标题名
* @param headers 表格属性列名数组
* @param dataSet 需要显示的数据集合,集合中一定要放置符合javabean风格的类的对象。此方法支持的
* javabean属性的数据类型有基本数据类型及String,Date,byte[](图片数据)
* @param pattern 如果有时间数据,设定输出格式。默认为"yyy-MM-dd"
*/
@SuppressWarnings({"unchecked", "rawtypes"})
public void exportExcel(String title, String[] headers,
Collection<T> dataSet, OutputStream out, String pattern) {
// 声明一个工作薄,将1000条写入内存,其余在硬盘,以防止内存溢出
SXSSFWorkbook workbook = new SXSSFWorkbook(1000);
// 生成一个表格
Sheet sheet = workbook.createSheet(title);
// 设置表格默认列宽度为10个字节
sheet.setDefaultColumnWidth((short) 10);
//设置表格头样式
CellStyle style = workbook.createCellStyle();
style.setAlignment(HorizontalAlignment.CENTER);//水平居中
style.setVerticalAlignment(VerticalAlignment.CENTER);//垂直居中
style.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.index);//设置背景色
style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
//设置边框
style.setBorderBottom(BorderStyle.THIN);
style.setBorderLeft(BorderStyle.THIN);
style.setBorderRight(BorderStyle.THIN);
style.setBorderTop(BorderStyle.THIN);
style.setAlignment(HorizontalAlignment.CENTER);
// 生成一个字体
Font font = workbook.createFont();
font.setFontName("黑体");
font.setFontHeightInPoints((short) 12);
// 把字体应用到当前的样式
style.setFont(font);
// 产生表格标题行
Row row = sheet.createRow(0);
for (short i = 0; i < headers.length; i++) {
Cell cell = row.createCell(i);
cell.setCellStyle(style);
XSSFRichTextString text = new XSSFRichTextString(headers[i]);
cell.setCellValue(text);
}
// 设置表格内容样式
CellStyle style2 = workbook.createCellStyle();
style2.setFillPattern(FillPatternType.SOLID_FOREGROUND);
style2.setBorderBottom(BorderStyle.THIN);
style2.setBorderLeft(BorderStyle.THIN);
style2.setBorderRight(BorderStyle.THIN);
style2.setBorderTop(BorderStyle.THIN);
style2.setAlignment(HorizontalAlignment.CENTER);
style2.setVerticalAlignment(VerticalAlignment.CENTER);
// 生成另一个字体
Font font2 = workbook.createFont();
font.setFontName("仿宋_GB2312");
font.setColor((short) 64);//黑色
// 把字体应用到当前的样式
style2.setFont(font2);
// 遍历集合数据,产生数据行
Iterator<T> it = dataSet.iterator();
int index = 0;
while (it.hasNext()) {
index++;
row = sheet.createRow(index);
T t = (T) it.next();
// 利用反射,根据javabean属性的先后顺序,动态调用getXxx()方法得到属性值
Field[] fields = t.getClass().getDeclaredFields();
for (short i = 0; i < fields.length; i++) {
Cell cell = row.createCell(i);
cell.setCellStyle(style2);
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[]{});
// 根据javabean的方法类型,判断值的类型后进行强制类型转换(具体需要自己根据情况修改)
String textValue = null;
//javabean里边的get和set方法为Integer时做处理
if (value instanceof Integer) {
int intValue = (Integer) value;
cell.setCellValue(intValue);
}
else if (value instanceof Double) {
double intValue = (Double) value;
cell.setCellValue(intValue);
}
else if (value instanceof Boolean) {
boolean bValue = (Boolean) value;
textValue = bValue ? "是" : "否";
cell.setCellValue(textValue);
}
else if (value instanceof Date) {
Date date = (Date) value;
SimpleDateFormat sdf = new SimpleDateFormat(pattern);
textValue = sdf.format(date);
cell.setCellValue(textValue);
} else if (value instanceof byte[]) {
// 声明一个画图的*管理器
Drawing drawing= sheet.createDrawingPatriarch();
row.setHeightInPoints(60);
sheet.setColumnWidth(i, (short) (35.7 * 80));
byte[] bsValue = (byte[]) value;
XSSFClientAnchor anchor = new XSSFClientAnchor(0, 0, 1023, 255, i, index, i+1, index+1);
anchor.setAnchorType(ClientAnchor.AnchorType.byId(0));
drawing.createPicture(anchor, workbook.addPicture(bsValue, XSSFWorkbook.PICTURE_TYPE_JPEG));
} else {
if (value != null) {
textValue = value.toString();
} else {
textValue = "";
}
cell.setCellValue(textValue);
}
} catch (SecurityException | NoSuchMethodException | IllegalArgumentException | InvocationTargetException | IllegalAccessException e) {
e.printStackTrace();
}
}
}
try {
workbook.write(out);
} catch (IOException e) {
e.printStackTrace();
}
}
}
商品实体类
package com.pec.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.Date;
@Data
@AllArgsConstructor
@NoArgsConstructor
public class Goods {
private Integer goodsID;
private String goodsName;
private Double price;
private Date effectiveDate;
private byte[] image;
private Boolean show;
}
excel导出工具类测试
@Test
void poiExport(){
try {
OutputStream os = new FileOutputStream(new File("D://test1.xlsx"));
List<Goods> list = new ArrayList<>();
ByteArrayOutputStream bos = new ByteArrayOutputStream();
FileInputStream fis = new FileInputStream(new File("D://12.jpg"));
int j;
byte[] data = new byte[1024];
while ((j = fis.read(data))!=-1){
bos.write(data,0,j);
}
fis.close();
bos.close();
byte[] img = bos.toByteArray();
for(int i = 0;i<10;i++){
Goods goods = new Goods(i,"商品"+i,20.500+i,new Date(),img,true);
list.add(goods);
}
String str[] = new String[]{"商品id","商品名称","商品价格","有效期","商品图片","是否显示"};
ExcelExport<Goods> export = new ExcelExport<>();
export.exportExcel("测试导出工具类",str,list,os,"yyyy-MM-dd HH:mm:ss");
} catch (IOException e) {
e.printStackTrace();
}
}
效果图
导入excel工具类
package com.pec.util;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Row;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.usermodel.Workbook;
import org.apache.poi.xssf.usermodel.XSSFWorkbook;
import java.io.InputStream;
import java.text.DecimalFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class ExcelImport {
/**
*
* @param fileName 文件
* @param is 文件流
* @param pattern 返回日期格式
* @param isExistTitle 是否存在表头
* @return map
*/
public Map<Integer, List<List<Object>>> getExcelRow(String fileName, InputStream is,String pattern, boolean isExistTitle){
List<List<Object>> list;
Map<Integer,List<List<Object>>> map = null;
Workbook workbook = null;
boolean isExcel03 = fileName.matches("^.+\\.(?i)(xls)$");
boolean isExcel07 = fileName.matches("^.+\\.(?i)(xlsx)$");
try {
if (isExcel03) {
workbook = new HSSFWorkbook(is);
}else if (isExcel07) {
workbook = new XSSFWorkbook(is);
}else{
return null;
}
int numberOfSheets = workbook.getNumberOfSheets();
map = new HashMap<>();
for (int i = 0; i < numberOfSheets; i++) {
Sheet sheetAt = workbook.getSheetAt(i);
int lastRowNum = sheetAt.getLastRowNum();//行数从0开始
list = new ArrayList<>();
for (int j = 0; j <= lastRowNum; j++) {
if (j == 0 && isExistTitle) {
j++;
continue;
}
Row row = sheetAt.getRow(j);
short lastCellNum = row.getLastCellNum();
List<Object> li = new ArrayList<>();
for (int k = 0; k < lastCellNum; k++) {
li.add(getCellValue(row.getCell(k), pattern));
}
list.add(li);
}
map.put(i, list);
}
} catch(Exception e){
e.printStackTrace();
}
return map;
}
private Object getCellValue(Cell cell,String pattern){
Object value = null;
DecimalFormat df = new DecimalFormat("0"); //格式化number String字符
SimpleDateFormat sdf = new SimpleDateFormat(pattern); //日期格式化
DecimalFormat df2 = new DecimalFormat("0.00"); //格式化数字
switch (cell.getCellType()) {
case STRING:
value = cell.getRichStringCellValue().getString();
break;
case NUMERIC:
if("General".equals(cell.getCellStyle().getDataFormatString())){
value = df.format(cell.getNumericCellValue());
}else if("m/d/yy".equals(cell.getCellStyle().getDataFormatString())){
value = sdf.format(cell.getDateCellValue());
}else{
value = df2.format(cell.getNumericCellValue());
}
break;
case BOOLEAN:
value = cell.getBooleanCellValue();
break;
case BLANK:
value = "";
break;
default:
break;
}
return value;
}
}
excel导入工具类测试
@Test
void importTest() {
FileInputStream fis;
ExcelImport excelImport = new ExcelImport();
try {
fis = new FileInputStream(new File("D://test.xls"));
Map<Integer,List<List<Object>>> map = excelImport.getExcelRow("test.xls", fis,"yyyy年MM月dd日",false);
if(map!=null) {
List<List<Object>> lists = map.get(0);
System.out.println(lists.toString());
}else{
System.out.println("导入excel文件格式错误");
}
} catch (Exception e) {
e.printStackTrace();
}
}