软件编程
位置:首页>> 软件编程>> java编程>> Springboot导出文件,前端下载文件方式

Springboot导出文件,前端下载文件方式

作者:coderping  发布时间:2023-07-21 11:27:05 

标签:Springboot,导出文件,前端,下载文件

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. 将文件以流的形式一次性读取到内存

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


/**
* @param path     想要下载的文件的路径
* @param response
* @功能描述 下载文件:
*/
@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. 将输入流中的数据循环写入到响应输出流中

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


/**
* @param path     指想要下载的文件的路径
* @param response
* @功能描述 下载文件:将输入流中的数据循环写入到响应输出流中,而不是一次性读取到内存
*/
@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. 下载网络文件到本地


/**
* @param path       下载后的文件路径和名称
* @param netAddress 文件所在网络地址
* @功能描述 网络文件下载到服务器本地
*/
@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. 网络文件获取到服务器后

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


/**
* @param netAddress
* @param filename
* @param isOnLine
* @param response
* @功能描述 网络文件获取到服务器后,经服务器处理后响应给前端
*/
@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

来源:https://blog.csdn.net/coderping/article/details/105491972

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com