导出excel通用模板

改良版,关于添加依赖之类的之前一篇文章里面有。
这篇是把之前的方法抽成通用模板。

一、添加一个实体类

package com.lencity.securitymanagementplatform.data.entity;

import java.util.List;

public class XlsData {

    public static final int DATA_TYPE_INTEGER = 0;
    public static final int DATA_TYPE_STRING = 1;
    private List<String> titles;//表头
    private List<Integer> types;//数据类型
    private List<List<Object>> values;存表数据
    
    public List<Integer> getTypes() {
        return types;
    }
    public void setTypes(List<Integer> types) {
        this.types = types;
    }
    public List<String> getTitles() {
        return titles;
    }
    public void setTitles(List<String> titles) {
        this.titles = titles;
    }
    public List<List<Object>> getValues() {
        return values;
    }
    public void setValues(List<List<Object>> values) {
        this.values = values;
    }   
}

二、创建一个service类

package com.lencity.securitymanagementplatform.service;

import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.List;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;
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.HSSFDataFormat;
import org.apache.poi.hssf.usermodel.HSSFFont;
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.ss.usermodel.HorizontalAlignment;
import org.springframework.stereotype.Service;

import com.lencity.securitymanagementplatform.data.entity.XlsData;

@Service
public class XlsService {

        //写一个接口,哪个控制器需要加上导出excel功能就继承这个接口
    public static interface IXlsOutputProcessor {
        public XlsData processXlsData(Map<String, Object> condition);
    }

        //解析数据创建excel
    public HSSFWorkbook createExcelData(IXlsOutputProcessor processor, Map<String, Object> condition) {
        XlsData xlsData = processor.processXlsData(condition);
        HSSFWorkbook workbook = new HSSFWorkbook();
        HSSFSheet sheet = workbook.createSheet("统计表");// 创建一个excel表单
        HSSFRow titleRow = sheet.createRow(0);
        // 设置列宽,setColumnWidth的第二个参数要乘以256,这个参数的单位是1/256个字符宽度
        sheet.setColumnWidth(1, 15 * 256);
        sheet.setColumnWidth(3, 20 * 256);

        HSSFCellStyle style = workbook.createCellStyle();
        style.setDataFormat(HSSFDataFormat.getBuiltinFormat("m/d/yy h:mm"));// 设置日期格式
        HSSFFont font = workbook.createFont();// 设置为居中加粗
        font.setBold(true);
        style.setAlignment(HorizontalAlignment.CENTER);
        style.setFont(font);
        List<String> titles = xlsData.getTitles();
        HSSFCell cell;
        /* 构造表头 */
        for (int i = 0; i < titles.size(); i++) {
            cell = titleRow.createCell(i);
            cell.setCellValue(titles.get(i));
            cell.setCellStyle(style);
        }
        int rowNum = 1;
        List<Integer> dataTypes = xlsData.getTypes();
        List<List<Object>> values = xlsData.getValues();
        for (int i = 0; i < values.size(); i++) {
            List<Object> value = values.get(i);
            HSSFRow row = sheet.createRow(rowNum);
            for (int j = 0; j < value.size(); j++) {
                switch (dataTypes.get(j)) {
                case XlsData.DATA_TYPE_INTEGER:
                    row.createCell(j).setCellValue((Integer) value.get(j));
                    break;
                case XlsData.DATA_TYPE_STRING:
                    row.createCell(j).setCellValue((String) value.get(j));
                    break;
                }
            }
            rowNum++;

        }
        return workbook;
    }


    // 浏览器导出excel
    public void buildExcelDocument(String filename, HSSFWorkbook workbook, HttpServletResponse response)
            throws Exception {
        response.reset();
        response.setContentType("application/vnd.ms-excel");
        response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(filename, "utf-8"));
        OutputStream outputStream = response.getOutputStream();
        workbook.write(outputStream);
        outputStream.flush();
        outputStream.close();
    }

    // 下载excel模板功能
    public void downloadTemplate(HttpServletResponse response,HttpServletRequest request) throws Exception {
        String fileName="导出模板.xls";
        response.reset();
        response.setContentType("application/vnd.ms-excel");
            response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "utf-8"));
            String filePath=request.getServletContext().getRealPath("/excel/")+fileName;
            FileInputStream input=new FileInputStream(filePath);
            OutputStream out=response.getOutputStream();
            byte[] b=new byte[2048];
            int len;
            while((len=input.read(b))!=-1) {
                out.write(b,0,len);
            }
             response.setHeader("Content-Length", String.valueOf(input.getChannel().size()));
                input.close();
    }
}

三、控制器
假设我们要在用户页面加上导出表格的功能,那就在用户的控制器上继承接口

public class UserController implements IXlsOutputProcessor {

继承之后需要在控制器中重写接口方法,


导出的表格样式.png
关于封装数据,主要就是根据自己实际的项目需求,来构造数据了
// 封装数据
    @Override
    public XlsData processXlsData(Map<String, Object> condition) {
        List<String> titles = new ArrayList<>();//表头
        List<Integer> dataTypes = new ArrayList<>();//表数据类型
        List<List<Object>> values = new ArrayList<>();//表头对应的数据
        titles.add("姓名");
        dataTypes.add(XlsData.DATA_TYPE_STRING);
        titles.add("手机号码");
        dataTypes.add(XlsData.DATA_TYPE_STRING);
        titles.add("职位");
        dataTypes.add(XlsData.DATA_TYPE_STRING);
        titles.add("部门");
        dataTypes.add(XlsData.DATA_TYPE_STRING);
        List<User> users = userService.getUsersByCondition(condition);
        XlsData xlsData = new XlsData();
        xlsData.setTitles(titles);
        xlsData.setTypes(dataTypes);
        for (User user : users) {
            List<Object> tmpList = new ArrayList<>();
            tmpList.add(user.getName());
            tmpList.add(user.getMobile());
            tmpList.add(user.getPosition());
            tmpList.add(departmentService.getDepartmentNameByDepartmentCode(user.getDepartmentCode()));
            values.add(tmpList);
        }
        xlsData.setValues(values);
        return xlsData;
    }

    // 导出excel,前台js,点击   导出excel   关联的路径就是这个
    @PostMapping(value = "/downLoadXls")
    @ResponseBody
    public String downLoadXls(Map<String, Object> condition, HttpServletResponse response) throws Exception {
        String fileName = "导出excel.xls";
        HSSFWorkbook workbook = xlsService.createExcelData(this, condition);
        xlsService.buildExcelDocument(fileName, workbook, response);
        JSONObject jsonObject = new JSONObject();
        jsonObject.put("code", 1);
        return jsonObject.toString();
    }
    // 下载模板,前台js,点击  下载模板   关联的路径就是这个
    @PostMapping(value = "/downloadTemplate")
    @ResponseBody
    public String downloadTemplate(HttpServletResponse response,HttpServletRequest request) throws Exception {
        String fileName = "导出excel.xls";
        xlsService.downloadTemplate(response, request);
        JSONObject jsonObject = new JSONObject();
        jsonObject.put("code", 1);
        return jsonObject.toString();
    }
前台界面.png

模板的存放位置

前台按钮代码

<button type="button" class="btn btn-primary waves-effect" onclick="downloadTemplate()" id="downloadTemplate">
<i class="material-icons">vertical_align_bottom</i> 
<span>下载模板</span>
</button>

<button type="button" class="btn btn-primary waves-effect"onclick="exportExcel()">
<i class="material-icons">vertical_align_bottom</i> 
<span>导出表格</span>
</button>

js

form表单里面是页面的表单筛选条件,如果要导数据库所有的数据,可把form表单去掉。如果导出的数据是有筛选条件的,需要改一下form表单
function exportExcel() {
   var name = $("#name").val();
   var departmentCode = $("#departmentCode").find("option:selected").val();
    var form = $("<form>");   
    $('body').append(form);  
        form.attr('style','display:none');   
        form.attr('target','');
        form.attr('method','post');
        form.attr('action',contextPath+'/user/downLoadXls');//下载文件的请求路径

        //对应查询条件的开始时间
        var input1 = $('<input>'); 
        input1.attr('type','hidden'); 
        input1.attr('name',"name"); 
        input1.attr('value',name);
        form.append(input1);  

        //对应查询条件的结束时间
        var input2 = $('<input>'); 
        input2.attr('type','hidden'); 
        input2.attr('name','departmentCode'); 
        input2.attr('value',departmentCode);
        form.append(input2);
        form.submit();    
}

下载模板的js

function downloadTemplate() {
    var form = $("<form>");
    $('body').append(form);
    form.attr('style', 'display:none');
    form.attr('target', '');
    form.attr('method', 'post');
    form.attr('action', contextPath + '/user/downloadTemplate');// 下载文件的请求路径
    form.submit();
}

Over~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,096评论 1 32
  • 你不会因为别人过得好而变得好,你更不会因为别人不好而变得好,所以啊我劝你善良
    爱你呦阿业阅读 229评论 0 1
  • 今天参加了抚仙湖国际马拉松,人生中第一次参加马拉松比赛,人生中第一个正式的马拉松比赛的十公里。 赛后,一个姐姐跟我...
    去看海Lily阅读 123评论 0 0
  • 到了晚上秦风临阵磨刀,想让婉茹帮他研究一下如何能够快速提升实力,好在今天的测试中通过。 婉茹听后面色古怪的看了...
    西土瓦大神阅读 522评论 0 3