文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Springboot如何导出文件

2023-06-21 22:02

关注

这篇文章主要介绍Springboot如何导出文件,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

后端代码

可以把请求设置为post,我这里是Get

 @RequestMapping(value = "/download", method = RequestMethod.POST)    public void download(HttpServletRequest request, HttpServletResponse res) throws Exception {        File excelFile = new File("/Users/i501695/GitHUbProject/EN_ProductIntergration/databaseclient/src/main/resources/Files/ProductTemplateCopy.xlsx");        res.setCharacterEncoding("UTF-8");        String realFileName = excelFile.getName();        res.setHeader("content-type", "application/octet-stream;charset=UTF-8");        res.setContentType("application/octet-stream;charset=UTF-8");        //加上设置大小下载下来的.xlsx文件打开时才不会报“Excel 已完成文件级验证和修复。此工作簿的某些部分可能已被修复或丢弃”        res.addHeader("Content-Length", String.valueOf(excelFile.length()));        try {            res.setHeader("Content-Disposition", "attachment;filename=" + java.net.URLEncoder.encode(realFileName.trim(), "UTF-8"));        } catch (UnsupportedEncodingException e1) {            e1.printStackTrace();        }        byte[] buff = new byte[1024];        BufferedInputStream bis = null;        OutputStream os = null;        try {            os = res.getOutputStream();            bis = new BufferedInputStream(new FileInputStream(excelFile));            int i = bis.read(buff);            while (i != -1) {                os.write(buff, 0, buff.length);                os.flush();                i = bis.read(buff);            }        }catch (Exception e){            e.printStackTrace();        }finally {            if (bis != null) {                try {                    bis.close();                } catch (IOException e) {                    e.printStackTrace();                }            }        }    }

前端伪代码结合Axios(核心代码一样,只是结合了Axios)

   Axios({ // 用axios发送post请求                method: 'post',                url: 'http://127.0.0.1:8762/dataService/download', // 请求地址                data: formData, // 参数                responseType: 'blob' // 表明返回服务器返回的数据类型            })                .then((res) => { // 处理返回的文件流                    let blob = new Blob([res.data], {type: res.data.type})                    const fileName = 'ProductTemplateCopy.xlsx';                    let downloadElement = document.createElement('a')                    let href = window.URL.createObjectURL(blob); //创建下载的链接                    downloadElement.href = href;                    downloadElement.download = fileName; //下载后文件名                    document.body.appendChild(downloadElement);                    downloadElement.click(); //点击下载                    document.body.removeChild(downloadElement); //下载完成移除元素                    window.URL.revokeObjectURL(href); //释放blob                    message.success('upload successfully.');            })            .catch(function (error) {                console.log(error);            });

SpringBoot文件下载的几种方式

1. 将文件以流的形式一次性读取到内存

通过响应输出流输出到前端

@RequestMapping("/download")public void download(String path, HttpServletResponse response) {    try {        // path是指想要下载的文件的路径        File file = new File(path);        log.info(file.getPath());        // 获取文件名        String filename = file.getName();        // 获取文件后缀名        String ext = filename.substring(filename.lastIndexOf(".") + 1).toLowerCase();        log.info("文件后缀名:" + ext);        // 将文件写入输入流        FileInputStream fileInputStream = new FileInputStream(file);        InputStream fis = new BufferedInputStream(fileInputStream);        byte[] buffer = new byte[fis.available()];        fis.read(buffer);        fis.close();        // 清空response        response.reset();        // 设置response的Header        response.setCharacterEncoding("UTF-8");        //Content-Disposition的作用:告知浏览器以何种方式显示响应返回的文件,用浏览器打开还是以附件的形式下载到本地保存        //attachment表示以附件方式下载   inline表示在线打开   "Content-Disposition: inline; filename=文件名.mp3"        // filename表示文件的默认名称,因为网络传输只支持URL编码的相关支付,因此需要将文件名URL编码后进行传输,前端收到后需要反编码才能获取到真正的名称        response.addHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(filename, "UTF-8"));        // 告知浏览器文件的大小        response.addHeader("Content-Length", "" + file.length());        OutputStream outputStream = new BufferedOutputStream(response.getOutputStream());        response.setContentType("application/octet-stream");        outputStream.write(buffer);        outputStream.flush();    } catch (IOException ex) {        ex.printStackTrace();    }}

2. 将输入流中的数据循环写入到响应输出流中

而不是一次性读取到内存,通过响应输出流输出到前端

@RequestMapping("/downloadLocal")public void downloadLocal(String path, HttpServletResponse response) throws IOException {   // 读到流中   InputStream inputStream = new FileInputStream(path);// 文件的存放路径   response.reset();   response.setContentType("application/octet-stream");   String filename = new File(path).getName();   response.addHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode(filename, "UTF-8"));   ServletOutputStream outputStream = response.getOutputStream();   byte[] b = new byte[1024];   int len;   //从输入流中读取一定数量的字节,并将其存储在缓冲区字节数组中,读到末尾返回-1   while ((len = inputStream.read(b)) > 0) {       outputStream.write(b, 0, len);   }   inputStream.close();}

3. 下载网络文件到本地

@RequestMapping("/netDownloadLocal")public void downloadNet(String netAddress, String path) throws IOException {  URL url = new URL(netAddress);  URLConnection conn = url.openConnection();  InputStream inputStream = conn.getInputStream();  FileOutputStream fileOutputStream = new FileOutputStream(path);  int bytesum = 0;  int byteread;  byte[] buffer = new byte[1024];  while ((byteread = inputStream.read(buffer)) != -1) {      bytesum += byteread;      System.out.println(bytesum);      fileOutputStream.write(buffer, 0, byteread);  }  fileOutputStream.close();}

4. 网络文件获取到服务器后

经服务器处理后响应给前端

@RequestMapping("/netDownLoadNet")public void netDownLoadNet(String netAddress, String filename, boolean isOnLine, HttpServletResponse response) throws Exception {    URL url = new URL(netAddress);    URLConnection conn = url.openConnection();    InputStream inputStream = conn.getInputStream();    response.reset();    response.setContentType(conn.getContentType());    if (isOnLine) {        // 在线打开方式 文件名应该编码成UTF-8        response.setHeader("Content-Disposition", "inline; filename=" + URLEncoder.encode(filename, "UTF-8"));    } else {        //纯下载方式 文件名应该编码成UTF-8        response.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode(filename, "UTF-8"));    }    byte[] buffer = new byte[1024];    int len;    OutputStream outputStream = response.getOutputStream();    while ((len = inputStream.read(buffer)) > 0) {        outputStream.write(buffer, 0, len);    }    inputStream.close();}

5. 常见异常和问题

(1)响应对象无需通过return返回

Springboot如何导出文件

原因: 响应对象是可以不用作为方法返回值返回的,其在方法执行时已经开始输出,且其无法与@RestController配合,以JSON格式返回给前端

Springboot如何导出文件

Springboot如何导出文件

解决办法: 删除return语句

(2)返回前端的文件名必须进行URL编码

原因: 网络传输只能传输特定的几十个字符,需要将汉字、特殊字符等经过Base64等编码来转化为特定字符,从而进行传输,而不会乱码

URLEncoder.encode(fileName, "UTF-8")
(3)IO流有待学习

1:read() :

从输入流中读取数据的下一个字节,返回0到255范围内的int字节值。如果因为已经到达流末尾而没有可用的字节,则返回-1。在输入数据可用、检测到流末尾或者抛出异常前,此方法一直阻塞。

2:read(byte[] b) :

从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中。以整数形式返回实际读取的字节数。在输入数据可用、检测到文件末尾或者抛出异常前,此方法一直阻塞。如果 b 的长度为 0,则不读取任何字节并返回 0;否则,尝试读取至少一个字节。如果因为流位于文件末尾而没有可用的字节,则返回值 -1

以上是“Springboot如何导出文件”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注编程网行业资讯频道!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯