如何实现前端异步调用导出Excel?

生来就可爱ヽ(ⅴ<●) 提交于 2020-04-28 15:11:45

今天分享JBolt极速开发平台中使用的异步下载文件的解决方案。

需求场景:

列表查询界面有一个【导出Excel】按钮,需要点击按钮,弹出loading信息框,异步调用后台action,得到数据库数据后生成Excel数据然后响应发送到前端,异步请求拿到数据后,当做文件下载下来。

重点:

异步请求,导出Excel,前端拿到后做成文件下载

 

解决方案:

1、前端异步请求

点击按钮的时候,执行一个异步请求,去访问Action

<button type="button" onclick="ajaxDownload('admin/user/exportExcel')">导出Excel</button>
//异步下载
function ajaxDownload(url,fileName){
 
    var xhr = new XMLHttpRequest();
    xhr.timeout=jboltAjaxTimeout;
    xhr.open('GET', url, true); 
    xhr.responseType = "blob";    // 返回类型blob
    //定义请求完成的处理函数
    xhr.onload = function () {
        // 请求完成
        if (this.status === 200) {
            // 返回200
            var blob = this.response;
            var reader = new FileReader();
            reader.readAsDataURL(blob);    // 转换为base64,可以直接放入a href
            reader.onload = function (e) {
                // 转换完成,创建一个a标签用于下载
                var a = document.createElement('a');
                if(fileName){
                    a.download =fileName;
                }else{
                    var headerName=xhr.getResponseHeader("Content-disposition");
                    if(headerName){
                        headerName=decodeURI(headerName);
                        var dl_filenameArr=headerName.split("=");
                        if(dl_filenameArr&&dl_filenameArr.length==2){
                            a.download =$.trim(dl_filenameArr[1]);
                        }
                    }else{
                        alert("下载失败,未设置下载文件的filename");
                        return false;
                    }
                }
                a.href = e.target.result;
                jboltBody.append(a);    // 修复firefox中无法触发click
                a.click();
                $(a).remove();
            }
        }else{
           alert("网络异常");
        }
    };
    // 发送ajax请求
    xhr.send();
}

 

好,这样前端在点击按钮的时候就会执行异步请求,去请求指定的URL地址。

那么后端就需要处理这个请求,从数据库查询后做成excel。

 

2、后端写出Excel文件流

注意:为了方便演示 看清楚后端导出流程 我没有再service里写 直接在action中快速写完逻辑。

大体意思就是读取数据,把生成的Excel文件流形式写出给Response。

其中header中的content-disposition属性值里的filename是自定义的。

前端按钮上调用ajaxDownload如果传入一个名字的话,会按照传入的名字去下载生成文件。具体是前端工作了。

/**
 * 导出Excel
 */
public void exportExcel() {
HttpServletResponse response=getResponse();
response.setHeader("Content-Disposition", "attachment;filename=jbolt.xls");
response.setHeader("Pragma", "no-cache");
response.setHeader("Cache-Control", "no-cache");
response.setHeader("Accept-Ranges", "bytes");
response.setDateHeader("Expires", 0);
response.setContentType("application/vnd.ms-excel");
//使用hutool 导出excel
ExcelWriter excelWriter=ExcelUtil.getWriter();
List<User> users=service.findAll();
int row=0;
//循环遍历
for(User user:users) {
excelWriter.writeCellValue(0,row, user.getId());
excelWriter.writeCellValue(1,row, user.getName());
excelWriter.writeCellValue(2,row, user.getPinyin());
row++;
}
//导出下载文件流
try {
excelWriter.flush(response.getOutputStream());
} catch (IORuntimeException | IOException e) {
e.printStackTrace();
}
excelWriter.close();
}

看一下效果:

image.png

image.png

这个按钮的高级延伸版,我在JBolt极速开发平台里实现的,后端做了更完善的处理,大体原理就是这样。

http://jbolt.cn/jbolt.html

可以登录演示版,找到这个异步按钮,点击感受一下!

 

image.png

 

有问题就加我微信

微信:mumengmeng

image.png

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!