如题,编程语言为java,利用Apache poi在服务器端生成带有数据和图片的excel的数据流至浏览器供其下载至本地,基本就是类似一个网站的数据导出功能。
基本主要代码是参考以上的网址的,本人在此基础上做了一些小的修改,就是使导出的图片呈比例的展示以及我本身的图片是从ftp上获取下来的,因此在插入的时候转换成字节流插入就好,还有就是自己根据自己所需生成表格的格式做些修改,这个就不明讲,自己看代码就好。
先说步骤:
1、把自己所要生成excel表格的数据统一存放进List表中,我这边是放一个实体类
2、在前端请求导出数据的时候, 再用自己已经生成号的List表数据去调用生成excel表格数据流传输至前方即可
3、前端最好用GET请求后台,这样前端可以直接调用后台该接口即可在浏览器完成下载,至于POST方法,本人没试成功,具体原因也不清楚。。。。
废话少说,show you code:
(1)设置excel的类
import java.awt.image.BufferedImage;
import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.List;
import javax.imageio.ImageIO;
import javax.servlet.http.HttpServletResponse;
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 cn.recour.wiicop.db.entity.JYZTVio;//需要导出数据的实体类
/**
*
* @类名: ExportToExcelUtil
* @描述: 设置excel'的信息
*
*/
public class ExportToExcelUtil {
public static void out(HttpServletResponse response, List<JYZTVio> list,String fileName)
throws Exception {
/**
* 指定下载名
*/
// String fileName = "违停拍数据.xls";
/**
* 编码
*/
fileName = new String(fileName.getBytes("UTF-8"), "UTF-8");
/**
* 去除空白行
*/
response.reset();
/**
* 指定下载的文件名与设置相应头
*/
response.setHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE");
response.setHeader("Access-Control-Max-Age", "3600");
response.setHeader("Access-Control-Allow-Headers", "x-requested-with,Authorization");
response.setHeader("Access-Control-Allow-Credentials", "true");
response.setCharacterEncoding("utf-8");
response.setHeader("Content-Disposition", "attachment;filename="
+ fileName);
/**
* 下载文件类型
*/
response.setContentType("application/vnd.ms-excel");
/**
* 别给我缓存
*/
response.setHeader("Pragma", "no-cache");
response.setHeader("Cache-Control", "no-cache");
response.setDateHeader("Expires", 0);
/**
* 得到输出流,放到缓冲区
*/
OutputStream output = response.getOutputStream();
BufferedOutputStream bufferedOutPut = new BufferedOutputStream(output);
/**
* 下面是导出的excel格式的设置
*/
/*******************我是分割线**************************************/
// 定义单元格报头
String worksheetTitle = "违停拍数据";
HSSFWorkbook wb = new HSSFWorkbook();
// 创建单元格样式
HSSFCellStyle cellStyleTitle = wb.createCellStyle();
// 指定单元格居中对齐
cellStyleTitle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
// 指定单元格垂直居中对齐
cellStyleTitle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);
// 指定当单元格内容显示不下时自动换行
cellStyleTitle.setWrapText(true);
HSSFCellStyle cellStyle = wb.createCellStyle();
// 指定单元格居中对齐
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER);
// 指定单元格垂直居中对齐
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);
// 指定当单元格内容显示不下时自动换行
cellStyle.setWrapText(true);
// 设置单元格字体
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontName("宋体");
font.setFontHeight((short) 200);
cellStyleTitle.setFont(font);
/****************wo是分割线*****************************************/
/**
* 工作表列名
*/
String id = "序号";
String cphm = "车牌号码";
String location = "违停地址";
String time = "违停时间";
String img = "违停照片";
HSSFSheet sheet = wb.createSheet();
ExportExcelUtil exportExcel = new ExportExcelUtil(wb, sheet);
/**
* 创建报表头部
*/
exportExcel.createNormalHead(worksheetTitle, 4);
// 定义第一行
HSSFRow row1 = sheet.createRow(1);
HSSFCell cell1 = row1.createCell(0);
/**
* 第一行第一列
*/
cell1.setCellStyle(cellStyleTitle);
cell1.setCellValue(new HSSFRichTextString(id));
/**
* 第一行第二列
*/
cell1 = row1.createCell(1);
cell1.setCellStyle(cellStyleTitle);
cell1.setCellValue(new HSSFRichTextString(cphm));
/**
* 第一行第三列
*/
cell1 = row1.createCell(2);
cell1.setCellStyle(cellStyleTitle);
cell1.setCellValue(new HSSFRichTextString(location));
/**
* 第一行第四列
*/
cell1 = row1.createCell(3);
cell1.setCellStyle(cellStyleTitle);
cell1.setCellValue(new HSSFRichTextString(time));
/**
* 第一行第五列
*/
cell1 = row1.createCell(4);
cell1.setCellStyle(cellStyleTitle);
cell1.setCellValue(new HSSFRichTextString(img));
/********************我是分割线*********************************/
/**
* 定义第二行,就是我们的数据
*/
HSSFRow row = sheet.createRow(2);
HSSFCell cell = row.createCell(1);
sheet.setColumnWidth(4, 30 * 256);//固定图片宽度大小为30*256,然后在下面设置图片以此宽度为基准设置高度
/**
* 遍历我们传进来的jyztVio-list
*/
JYZTVio jyztVio =null;
HSSFPatriarch patriarch = sheet.createDrawingPatriarch();
for (int i = 0; i < list.size(); i++) {
jyztVio = list.get(i);
/**
* 从i+2行开始,因为我们之前的表的标题和列的标题已经占用了两行
*/
row = sheet.createRow(i + 2);
int s = i+1;
//...id
cell = row.createCell(0);
cell.setCellStyle(cellStyle);
cell.setCellValue(new HSSFRichTextString(s + ""));
//....车牌号码cphm
cell = row.createCell(1);
cell.setCellStyle(cellStyle);
cell.setCellValue(new HSSFRichTextString("\n"+jyztVio.getCphm()+"\n"));
//....违停地址
cell = row.createCell(2);
cell.setCellStyle(cellStyle);
cell.setCellValue(new HSSFRichTextString("\n"+jyztVio.getLocation()+"\n"));
//....违停时间
cell = row.createCell(3);
cell.setCellStyle(cellStyle);
cell.setCellValue(new HSSFRichTextString("\n"+jyztVio.getTime()+"\n"));
//...违停拍照
short h =2;//定义图片所在行
short l=4;//定义图片所在列
cell = row.createCell(4);
cell.setCellStyle(cellStyle);
//得到图片 new FileUtils().getFile(String path)这是我从ftp上获取图片的方法,里面参数是图片路径,各位具体情况具体处理
InputStream sbs = new ByteArrayInputStream(new FileUtils().getFile(jyztVio.getImage()));
BufferedImage image = ImageIO.read(sbs);
ByteArrayOutputStream byteArrayOut = new ByteArrayOutputStream();
int width = image.getWidth();//图片原始宽度
int height = image.getHeight();//图片原始高度
//根据宽度按比例设置高度
height = (int) Math.round((height * (30 * 13) * 1.0 / width));
row.setHeight((short) (height / 2 * 20));
ImageIO.write(image, "jpg", byteArrayOut);
HSSFClientAnchor anchor = new HSSFClientAnchor(0, 0, 0, 0, l, i + h, (short) (l+1), (i+1) + h);
anchor.setAnchorType(3);
// 插入图片
patriarch.createPicture(anchor, wb.addPicture(
byteArrayOut.toByteArray(),
HSSFWorkbook.PICTURE_TYPE_JPEG));
}
try {
/**
* 输出到浏览器
*/
bufferedOutPut.flush();
wb.write(bufferedOutPut);
bufferedOutPut.close();//关流
} catch (IOException e) {
e.printStackTrace();
} finally {
list.clear();
}
}
}
(2)excel导出的工具类
/**
* @author czy
* @version 创建时间:2017年10月31日 下午5:37:17
*
*/
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFCellStyle;
import org.apache.poi.hssf.usermodel.HSSFFont;
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;
import org.apache.poi.hssf.util.Region;
/**
*
* @类名: ExportExcel
* @描述: excel工具类
*
*/
public class ExportExcelUtil {
private HSSFWorkbook wb = null;
private HSSFSheet sheet = null;
/**
*
* <p>
* 标题:
* </p>
* <p>
* 描述:
* </p>
*
* @param wb
* @param sheet
*/
public ExportExcelUtil(HSSFWorkbook wb, HSSFSheet sheet) {
// super();
this.wb = wb;
this.sheet = sheet;
}
/**
*
* @标题: createNormalHead
* @描述: TODO(这里用一句话描述这个方法的作用)
* @参数: @param headString 头部的字符
* @参数: @param colSum 报表的列数
* @返回: void 返回类型
* @抛出:
*/
@SuppressWarnings("deprecation")
public void createNormalHead(String headString, int colSum) {
HSSFRow row = sheet.createRow(0);
// 设置第一行
HSSFCell cell = row.createCell(0);
// row.setHeight((short) 1000);
// 定义单元格为字符串类型
cell.setCellType(HSSFCell.ENCODING_UTF_16);// 中文处理
cell.setCellValue(new HSSFRichTextString(headString));
// 指定合并区域
/**
* public Region(int rowFrom, short colFrom, int rowTo, short colTo)
*/
sheet.addMergedRegion(new Region(0, (short) 0, 0, (short) colSum));
// 定义单元格格式,添加单元格表样式,并添加到工作簿
HSSFCellStyle cellStyle = wb.createCellStyle();
// 设置单元格水平对齐类型
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 指定单元格居中对齐
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);// 指定单元格垂直居中对齐
cellStyle.setWrapText(true);// 指定单元格自动换行
// 设置单元格字体
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontName("宋体");
font.setFontHeight((short) 600);
cellStyle.setFont(font);
cell.setCellStyle(cellStyle);
}
/**
*
* @标题: createNormalTwoRow
* @描述: 创建通用报表第二行
* @参数: @param params 二行统计条件数组
* @参数: @param colSum 需要合并到的列索引
* @返回: void 返回类型
* @抛出:
*/
@SuppressWarnings("deprecation")
public void createNormalTwoRow(String[] params, int colSum) {
// 创建第二行
HSSFRow row1 = sheet.createRow(1);
row1.setHeight((short) 400);
HSSFCell cell2 = row1.createCell(0);
cell2.setCellType(HSSFCell.ENCODING_UTF_16);
cell2.setCellValue(new HSSFRichTextString("时间:" + params[0] + "至"
+ params[1]));
// 指定合并区域
sheet.addMergedRegion(new Region(1, (short) 0, 1, (short) colSum));
HSSFCellStyle cellStyle = wb.createCellStyle();
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 指定单元格居中对齐
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);// 指定单元格垂直居中对齐
cellStyle.setWrapText(true);// 指定单元格自动换行
// 设置单元格字体
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontName("宋体");
font.setFontHeight((short) 250);
cellStyle.setFont(font);
cell2.setCellStyle(cellStyle);
}
/**
*
* @标题: createColumHeader
* @描述: 设置报表标题
* @参数: @param columHeader 标题字符串数组
* @返回: void 返回类型
* @抛出:
*/
public void createColumHeader(String[] columHeader) {
// 设置列头 在第三行
HSSFRow row2 = sheet.createRow(2);
// 指定行高
row2.setHeight((short) 600);
HSSFCellStyle cellStyle = wb.createCellStyle();
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 指定单元格居中对齐
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);// 指定单元格垂直居中对齐
cellStyle.setWrapText(true);// 指定单元格自动换行
// 单元格字体
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontName("宋体");
font.setFontHeight((short) 250);
cellStyle.setFont(font);
// 设置单元格背景色
cellStyle.setFillForegroundColor(HSSFColor.GREY_25_PERCENT.index);
cellStyle.setFillPattern(HSSFCellStyle.SOLID_FOREGROUND);
HSSFCell cell3 = null;
for (int i = 0; i < columHeader.length; i++) {
cell3 = row2.createCell(i);
cell3.setCellType(HSSFCell.ENCODING_UTF_16);
cell3.setCellStyle(cellStyle);
cell3.setCellValue(new HSSFRichTextString(columHeader[i]));
}
}
/**
*
* @标题: cteateCell
* @描述: 创建内容单元格
* @参数: @param wb HSSFWorkbook
* @参数: @param row HSSFRow
* @参数: @param col short型的列索引
* @参数: @param align 对齐方式
* @参数: @param val 列
* @返回: void 返回类型
* @抛出:
*/
public void cteateCell(HSSFWorkbook wb, HSSFRow row, int col, short align,
String val) {
HSSFCell cell = row.createCell(col);
cell.setCellType(HSSFCell.ENCODING_UTF_16);
cell.setCellValue(new HSSFRichTextString(val));
HSSFCellStyle cellstyle = wb.createCellStyle();
cellstyle.setAlignment(align);
cell.setCellStyle(cellstyle);
}
/**
*
* @标题: createLastSumRow
* @描述: 创建合计行
* @参数: @param colSum 需要合并到的列索引
* @参数: @param cellValue 设定文件
* @返回: void 返回类型
* @抛出:
*/
@SuppressWarnings("deprecation")
public void createLastSumRow(int colSum, String[] cellValue) {
HSSFCellStyle cellStyle = wb.createCellStyle();
cellStyle.setAlignment(HSSFCellStyle.ALIGN_CENTER); // 指定单元格居中对齐
cellStyle.setVerticalAlignment(HSSFCellStyle.VERTICAL_CENTER);// 指定单元格垂直居中对齐
cellStyle.setWrapText(true);// 指定单元格自动换行
// 单元格字体
HSSFFont font = wb.createFont();
font.setBoldweight(HSSFFont.BOLDWEIGHT_BOLD);
font.setFontName("宋体");
font.setFontHeight((short) 250);
cellStyle.setFont(font);
// 获取工作表最后一行
HSSFRow lastRow = sheet.createRow((short) (sheet.getLastRowNum() + 1));
HSSFCell sumCell = lastRow.createCell(0);
sumCell.setCellValue(new HSSFRichTextString("合计"));
sumCell.setCellStyle(cellStyle);
// 合并 最后一行的第零列-最后一行的第一列
sheet.addMergedRegion(new Region(sheet.getLastRowNum(), (short) 0,
sheet.getLastRowNum(), (short) colSum));// 指定合并区域
for (int i = 2; i < (cellValue.length + 2); i++) {
// 定义最后一行的第三列
sumCell = lastRow.createCell(i);
sumCell.setCellStyle(cellStyle);
// 定义数组 从0开始。
sumCell.setCellValue(new HSSFRichTextString(cellValue[i - 2]));
}
}
/**
*
* @标题: outputExcel
* @描述: 输出excel下载
* @参数: @param fileName 文件名
* @返回: void 返回类型
* @抛出:
*/
public void outputExcel(String fileName) {
FileOutputStream fos = null;
try {
fos = new FileOutputStream(new File(fileName));
wb.write(fos);
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
*
* @返回: HSSFSheet 返回类型
*/
public HSSFSheet getSheet() {
return sheet;
}
public void setSheet(HSSFSheet sheet) {
this.sheet = sheet;
}
/**
*
* @返回: HSSFWorkbook 返回类型
*/
public HSSFWorkbook getWb() {
return wb;
}
/**
*
*/
public void setWb(HSSFWorkbook wb) {
this.wb = wb;
}
}
Controller层的代码:
/**
* 下载违停拍数据
* @param body
* @param request
* @param response
*/
@RequestMapping(value = "/downloaddata", method = RequestMethod.GET)
public @ResponseBody void downloaddata(HttpServletRequest request,HttpServletResponse response){
try {
String fileName = request.getParameter("fileName");
List<JYZTVio> jyztViolist = jyztVioService.downloaddata(request,response);
if(jyztViolist==null){
preJson(response,"无下载内容");
}else{
ExportToExcelUtil.out(response, jyztViolist, fileName);
}
} catch (Exception e) {
e.printStackTrace();
}
}
这样,前端只要调用该接口,就可以在浏览器完成下载功能。
注意:本人以上生成excel表和Excel工具类的代码基本是借鉴以上网址而来,只是根据自己功能需求做了一些改变,并没有想拿来做商业用途,如需要可删!