当前位置:主页 > 查看内容

Spring Boot + Vue前后端分离,两种文件上传方式总结!

发布时间:2021-06-29 00:00| 位朋友查看

简介:在Vue.js 中,如果网络请求使用 axios ,并且使用了 ElementUI 库,那么一般来说,文件上传有两种不同的实现方案: 通过 Ajax 实现文件上传 通过 ElementUI 里边的 Upload 组件实现文件上传 两种方案,各有优缺点,我们分别来看。 准备工作 首先我们需要一点……

在Vue.js 中,如果网络请求使用 axios ,并且使用了 ElementUI 库,那么一般来说,文件上传有两种不同的实现方案

  1.  通过 Ajax 实现文件上传
  2.  通过 ElementUI 里边的 Upload 组件实现文件上传

两种方案,各有优缺点,我们分别来看。

准备工作

首先我们需要一点点准备工作,就是在后端提供一个文件上传接口,这是一个普通的 Spring Boot 项目,如下:

  1. SimpleDateFormat sdf = new SimpleDateFormat("/yyyy/MM/dd/");  
  2. @PostMapping("/import")  
  3. public RespBean importData(MultipartFile file, HttpServletRequest req) throws IOException {  
  4.     String format = sdf.format(new Date());  
  5.     String realPath = req.getServletContext().getRealPath("/upload") + format;  
  6.     File folder = new File(realPath);  
  7.     if (!folder.exists()) {  
  8.         folder.mkdirs();  
  9.     }  
  10.     String oldName = file.getOriginalFilename();  
  11.     String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf("."));  
  12.     file.transferTo(new File(folder,newName));  
  13.     String url = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/upload" + format + newName;  
  14.     System.out.println(url);  
  15.     return RespBean.ok("上传成功!");  

这里的文件上传比较简单,上传的文件按照日期进行归类,使用 UUID 给文件重命名。

这里为了简化代码,我省略掉了异常捕获,上传结果直接返回成功,后端代码大伙可根据自己的实际情况自行修改。

Ajax 上传

在 Vue 中,通过 Ajax 实现文件上传,方案和传统 Ajax 实现文件上传基本上是一致的,唯一不同的是查找元素的方式。

  1. <input type="file" ref="myfile">  
  2. <el-button @click="importData" type="success" size="mini" icon="el-icon-upload2">导入数据</el-button> 

在这里,首先提供一个文件导入 input 组件,再来一个导入按钮,在导入按钮的事件中来完成导入的逻辑。

  1. importData() {  
  2.   let myfile = this.$refs.myfile;  
  3.   let files = myfile.files;  
  4.   let file = files[0];  
  5.   var formData = new FormData();  
  6.   formData.append("file", file);  
  7.   this.uploadFileRequest("/system/basic/jl/import",formData).then(resp=> 
  8.     if (resp) {  
  9.       console.log(resp);  
  10.     }  
  11.   })  

关于这段上传核心逻辑,解释如下:

  1.  首先利用 Vue 中的 $refs 查找到存放文件的元素。
  2.  type 为 file 的 input 元素内部有一个 files 数组,里边存放了所有选择的 file,由于文件上传时,文件可以多选,因此这里拿到的 files 对象是一个数组。
  3.  从 files 对象中,获取自己要上传的文件,由于这里是单选,所以其实就是数组中的第一项。
  4.  构造一个 FormData ,用来存放上传的数据,FormData 不可以像 Java 中的 StringBuffer 使用链式配置。
  5.  构造好 FromData 后,就可以直接上传数据了,FormData 就是要上传的数据。
  6.  文件上传注意两点,1. 请求方法为 post,2. 设置 Content-Type 为  multipart/form-data 。

这种文件上传方式,实际上就是传统的 Ajax 上传文件,和大家常见的 jQuery 中写法不同的是,这里元素查找的方式不一样(实际上元素查找也可以按照JavaScript 中原本的写法来实现),其他写法一模一样。这种方式是一个通用的方式,和使用哪一种前端框架无关。最后再和大家来看下封装的上传方法:

  1. export const uploadFileRequest = (url, params) => {  
  2.   return axios({  
  3.     method: 'post',  
  4.     url: `${base}${url}`,  
  5.     data: params,  
  6.     headers: {  
  7.       'Content-Type': 'multipart/form-data'  
  8.     } 
  9.   });  

经过这几步的配置后,前端就算上传完成了,可以进行文件上传了。

使用 Upload 组件

如果使用 Upload ,则需要引入 ElementUI,所以一般建议,如果使用了 ElementUI 做 UI 控件的话,则可以考虑使用 Upload 组件来实现文件上传,如果没有使用 ElementUI 的话,则不建议使用 Upload 组件,至于其他的 UI 控件,各自都有自己的文件上传组件,具体使用可以参考各自文档。

  1. <el-upload  
  2.   style="display: inline"  
  3.   :show-file-list="false"  
  4.   :on-success="onSuccess"  
  5.   :on-error="onError"  
  6.   :before-upload="beforeUpload"  
  7.   action="/system/basic/jl/import">  
  8.   <el-button size="mini" type="success" :disabled="!enabledUploadBtn" :icon="uploadBtnIcon">{{btnText}}</el-button>  
  9. </el-upload> 
  1.  show-file-list 表示是否展示上传文件列表,默认为true,这里设置为不展示。
  2.  before-upload 表示上传之前的回调,可以在该方法中,做一些准备工作,例如展示一个进度条给用户 。
  3.  on-success 和 on-error 分别表示上传成功和失败时候的回调,可以在这两个方法中,给用户一个相应的提示,如果有进度条,还需要在这两个方法中关闭进度条。
  4.  action 指文件上传地址。
  5.  上传按钮的点击状态和图标都设置为变量 ,在文件上传过程中,修改上传按钮的点击状态为不可点击,同时修改图标为一个正在加载的图标 loading。
  6.  上传的文本也设为变量,默认上传 button 的文本是 数据导入 ,当开始上传后,将找个 button 上的文本修改为 正在导入。

相应的回调如下:

  1. onSuccess(response, file, fileList) {  
  2.   this.enabledUploadBtn = true 
  3.   this.uploadBtnIcon = 'el-icon-upload2' 
  4.   this.btnText = '数据导入' 
  5. },  
  6. onError(err, file, fileList) {  
  7.   this.enabledUploadBtn = true 
  8.   this.uploadBtnIcon = 'el-icon-upload2' 
  9.   this.btnText = '数据导入' 
  10. },  
  11. beforeUpload(file) {  
  12.   this.enabledUploadBtn = false 
  13.   this.uploadBtnIcon = 'el-icon-loading' 
  14.   this.btnText = '正在导入' 
  1.  在文件开始上传时,修改上传按钮为不可点击,同时修改上传按钮的图标和文本。
  2.  文件上传成功或者失败时,修改上传按钮的状态为可以点击,同时恢复上传按钮的图标和文本。

上传效果图如下:

总结

两种上传方式各有优缺点:

  1.  第一种方式最大的优势是通用,一招鲜吃遍天,到哪里都能用,但是对于上传过程的监控,进度条的展示等等逻辑都需要自己来实现。
  2.  第二种方式不够通用,因为它是 ElementUI 中的组件,得引入 ElementUI 才能使用,不过这种方式很明显有需多比较方便的回调,可以实现非常方便的处理常见的各种上传问题。
  3.  常规的上传需求第二种方式可以满足,但是如果要对上传的方法进行定制,则还是建议使用第一种上传方案。 
【责任编辑:庞桂玉 TEL:(010)68476606】
本文转载自网络,原文链接:https://mp.weixin.qq.com/s/_ddrk1Yl8rD33mc7IrJtow
本站部分内容转载于网络,版权归原作者所有,转载之目的在于传播更多优秀技术内容,如有侵权请联系QQ/微信:153890879删除,谢谢!

推荐图文

  • 周排行
  • 月排行
  • 总排行

随机推荐