通常前后端数据交互都是用JQuery的ajax函数,其返回类型只有xml、text、json、html等类型,没有“流”类型,所以我们无法使用ajax实现文件下载。下面介绍几种文件下载的思路。

一、windows.open下载文件

后端返回的是文件流

1.1 前端代码

var downloadURL = "appraise/download?flightNo=123";
window.open(downloadURL);

1.2 后端代码

response.setContentType("application/vnd.ms-excel");
response.setCharacterEncoding("utf-8");
// 这里URLEncoder.encode可以防止中文乱码 当然和easyexcel没有关系
String fileName = URLEncoder.encode("测试", "UTF-8");
//通知浏览器下载文件而不是打开
response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xls");

//工作薄
HSSFWorkbook workbook = new HSSFWorkbook();
int index = 0;
for(Map.Entry<String, List<EvaluationDo>> entry : evaluationTypeMap.entrySet()){
    if(providers.containsKey(entry.getKey())){
        AbstractEvaluationProvider evaluationDataProvider = providers.get(entry.getKey());
        ExportExcelUtils.exportExcel(workbook,index++,evaluationDataProvider.createSheet(),evaluationDataProvider.createHead(),evaluationDataProvider.createData(entry.getValue()));
    }
}
workbook.write(response.getOutputStream());

1.3 优点

  • 浏览器兼容性好

1.4 缺点

  • URL长度受限制
  • 拿不到后端处理这个过程的时机,无法根据回调函数做交互以及进度提示

二、ajax提交请求,后端返回文件在线地址

后端返回的是文件地址(文件地址可访问)

2.1 前端代码

$.ajax({
    type: "post",
    url: "appraise/download",
    data: {'flightNo':'123'},
    success: function (res) {
        if (res.Status) {
            // window.open或者a标签下载 
            var isSupportDownload = 'download' in document.createElement('a');
            if (isSupportDownload) {
                var $a = $("<a>");
                $a.attr({
                    href: res.url,
                    download: 'filename'
                }).hide().appendTo($("body"))[0].click();
            } else {
                window.open(res.url)
            }
        } else {
            alert(res.Message);
        }
    }
})

2.2 后端代码

return "doc/adscf-1123-221ss-dda.doc";

2.3 优点

  • 可以获取文件返回时机,可以做交互

2.4 缺点

  • 线上产生大量的中间临时文件,可以用设置时限来优化或可使用大厂的云存储,从而减少临时文件的产生

三、使用form.submit下载文件

后端返回文件流

3.1 前端代码

try{
        var exportForm = $("<form action='appraise/downLoad' method='post'></form>");
        exportForm.append("<input type='hidden' name='flightLeftDate' value='" + flightLeftDate + "'/>");
        exportForm.append("<input type='hidden' name='flightRightDate' value='" + flightRightDate + "'/>");
        exportForm.append("<input type='hidden' name='evaluationType' value='vectorDataEvaluation'/>");
        $(document.body).append(exportForm);
        exportForm.submit();
    }catch (e) {
        alert_prompt(e);
    }finally {
        exportForm.remove();
    }

3.2 后端代码

response.setContentType("application/vnd.ms-excel");
response.setCharacterEncoding("utf-8");
// 这里URLEncoder.encode可以防止中文乱码 当然和easyexcel没有关系
String fileName = URLEncoder.encode("测试", "UTF-8");
//通知浏览器下载文件而不是打开
response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xls");

//工作薄
HSSFWorkbook workbook = new HSSFWorkbook();
int index = 0;
for(Map.Entry<String, List<EvaluationDo>> entry : evaluationTypeMap.entrySet()){
    if(providers.containsKey(entry.getKey())){
        AbstractEvaluationProvider evaluationDataProvider = providers.get(entry.getKey());
        ExportExcelUtils.exportExcel(workbook,index++,evaluationDataProvider.createSheet(),evaluationDataProvider.createHead(),evaluationDataProvider.createData(entry.getValue()));
    }
}
workbook.write(response.getOutputStream());

3.3 优点

  • 兼容性良好,传统方式,不会出现URL长度限制问题

3.4 缺点

  • 拿不到后端处理这个过程的时机,无法根据回调函数做交互以及进度提示

四、使用 jquery-download 插件

  • jquery-download 下载地址:https://github.com/johnculviner/jquery.fileDownload/blob/master/src/Scripts/jquery.fileDownload.js
  • jquery-download CDN地址:https://www.bootcdn.cn/jquery.fileDownload/

4.1 前端代码

$.fileDownload('appraise/downLoad.jhtml', {
                httpMethod: 'post',
                data: {'flightLeftDate': flightLeftDate,'flightRightDate':flightRightDate},
                prepareCallback: function (url) {
                    console.log("文件下载中...");
                    // 数据加载动画
                    $("#loading").modal('show');
                },
                abortCallback: function (url) {
                    // 异常终止
                    alert_prompt("文件下载异常!");
                    $("#loading").modal('hide');
                },
                successCallback: function (url) {
                    alert_prompt("文件下载成功!");
                    $("#loading").modal('hide');
                },
                failCallback: function (html, url) {
                    if(html.indexOf('<') >= 0) {
                        html = $(html).text();
                    }
                    var result = JSON.parse(html);
                    $("#loading").modal('hide');
                    alert_prompt("文件下载失败:" + result.Head.Msg);
                }
            });

4.2 后端代码

if(!evaluationTypeMap.isEmpty()){
            try{
                response.setContentType("application/vnd.ms-excel");
                response.setCharacterEncoding("utf-8");
                // 这里URLEncoder.encode可以防止中文乱码 当然和easyexcel没有关系
                String fileName = URLEncoder.encode("测试", "UTF-8");
                response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xls");
                response.setHeader("Set-Cookie", "fileDownload=true; path=/");

                //工作薄
                HSSFWorkbook workbook = new HSSFWorkbook();
                int index = 0;
                for(Map.Entry<String, List<EvaluationDo>> entry : evaluationTypeMap.entrySet()){
                    if(providers.containsKey(entry.getKey())){
                        AbstractEvaluationProvider evaluationDataProvider = providers.get(entry.getKey());
                        ExportExcelUtils.exportExcel(workbook,index++,evaluationDataProvider.createSheet(),evaluationDataProvider.createHead(),evaluationDataProvider.createData(entry.getValue()));
                    }
                }
                workbook.write(response.getOutputStream());
            }catch (Exception e){
                log.error("efb:评价导出---->导出失败:",e);
                // 重置response
                response.reset();
                response.setContentType("application/json");
                response.setCharacterEncoding("utf-8");
                EfbReturnPO efbReturnPO = new EfbReturnPO();
                efbReturnPO.setUnSuccessHead("下载文件失败:" + e.getMessage());
                response.setHeader("Set-Cookie", "fileDownload=false; path=/");

                response.getWriter().println(AOSJson.toJson(efbReturnPO));
            }
        }else{
            log.error("efb:评价导出---->导出失败:无匹配数据");
            // 重置response
            response.reset();
            response.setContentType("application/json");
            response.setCharacterEncoding("utf-8");
            EfbReturnPO efbReturnPO = new EfbReturnPO();
            efbReturnPO.setUnSuccessHead("无匹配数据!");
            response.setHeader("Set-Cookie", "fileDownload=false; path=/");

            response.getWriter().println(AOSJson.toJson(efbReturnPO));
        }

注意:后端代码增加了一个名为"fileDownload"的cookie的返回;jquery.download.js插件使用该cookie来判断是否下载成功,从而进入成功回调函数(successCallback)

4.3 优点

  • 浏览器兼容好,此插件做了多种兼容
  • window.open(url)打开某个文件地址
  • iframe的框架中,设置src属性,通过iframe进行文件的下载,支持文件地址
  • 通过form标签,设置action的文件地址,然后通过form的提交来完成文件的下载
  • 可以获取文件返回时机,可以做交互

五、其它方案

  • iframe直接向后端提交,实现对文件流进行下载
  • Html5的Blob对象实现对文件流进行下载
  • file-saver实现对文件流进行下载