前端技术:vue,element-ui,axios

后台技术:springboot

本篇博客只给出关键的代码,提供思路,完全的涉及保密不方便提供

一,上传:

上传element组件代码,支持多文件,拖动文件进行上传:

<!--选择文件-->
                    <el-form-item label="选择模板" :label-width="formLabelWidth" :required="trueFlag">
                        <el-row>
                            <el-col :span="12">
                                <div class="grid-content bg-purple">
                                    <el-upload
                                            class="upload-demo"
                                            ref="upload"
                                            :action="actionUrl"
                                            :auto-upload="false"
                                            :limit="10"
                                            multiple
                                            accept="application/pdf"
                                            :before-upload="beforeUpload"
                                            :before-remove="beforeRemove"
                                            :on-remove="handleRemove"
                                            :on-change="fileChange"
                                            :file-list="fileList"
                                            :drag="trueFlag">
                                        <el-button size="small" type="primary">选取模板</el-button>
                                        <!--<el-button style="margin-left: 10px;" size="small" type="success" @click="submitUpload">上传到服务器</el-button>-->
                                        <div slot="tip" class="el-upload__tip">只能上传pdf文件,且不超过2M</div>
                                    </el-upload>
                                </div>
                            </el-col>
                        </el-row>
                    </el-form-item>
                </el-form>
                <div slot="footer" class="dialog-footer">
                    <el-button v-on:click="uploadDialogFormVisible = false">关闭</el-button>
                    <!--<el-button type="primary" v-on:click="uploadDialogFormVisible = false">上传</el-button>-->
                    <el-button type="primary" @click="submitUpload">上传到服务器</el-button>
                </div>
            </el-dialog>

 

上传前端js方法:

submitUpload: function () {
            if (this.area == null || this.area == '' || this.fileList.length < 1) {
                this.$message({
                    message: '区域未选择,或未选取上传模板',
                    type: 'error'
                });
                return;
            }
            let formData = new FormData();
            this.fileList.forEach(function (file) {
                formData.append("files", file.raw, file.name);
            });
            let config = {
                'Content-Type': 'multipart/form-data'
            };
            formData.append("area", this.area);
            let var_this = this;
            axios.post('/model/upload', formData, config)
                .then(function (response) {
                    if (response.data.status == 1) {
                        var_this.$message({
                            message: `${var_this.area}区域${var_this.fileList.length}个文件(上传/更新)成功`,
                            type: 'success'
                        });
                    } else {
                        var_this.$message({
                            message: response.data.msg,
                            type: 'error'
                        });
                    }
                }).catch(function (error) {
                console.log(error);
            })
        }

      说明:这里使用的是表单上传多文件,element-ui上传组件默认多文件上传实际上是一个文件一个文件进行上传的,这里利用表单可以实现一次请求上传多个文件,而且,上传文件的同时可以添加其他额外的属性,字段信息

 

后台springboot接口核心代码:

@PostMapping("upload")
    @ResponseBody
    public ResponseResult uploadModel(@ModelAttribute ModelFileVO modelFileVO) {
          //文件上传具体处理逻辑
        return rs;
    }

处理多文件上传的接口,内部处理逻辑网上很多资料,可以自己查找资料,主要使用到了@ModelAttribute注解,这个注解支持前端表单形式的参数,这里用来接收多文件,以及其他的参数,下面这个类是具体说明,出了多文件,还使用了额外的area字段,files表示多文件数组,@Data是用来解决set,get问题

 

 

@Data
public class ModelFileVO {
    private String area;
    private MultipartFile[] files;

}

 

效果图如下:

elementui安装应该在什么路径 elementui 下载文件_vue

 

 

elementui安装应该在什么路径 elementui 下载文件_springboot_02

 

二,下载:

element-ui下载核心代码:

<template slot-scope="scope">
                            <!--下载按钮-->
                            <el-button type="warning" icon="el-icon-download" :circle="isCircle" @click="downloadModel(scope.row)"></el-button>
                            <!--编辑按钮-->
                            <el-button type="primary" v-on:click="editUpload" icon="el-icon-edit"
                                       :circle="isCircle"></el-button>
                            <!--删除按钮-->
                            <el-button type="danger" v-on:click="deleteModel(scope.$index, scope.row)"
                                       icon="el-icon-delete" :circle="isCircle"></el-button>
                        </template>

 

下载vue.js的处理核心代码:

downloadModel: function (row) {
            console.log(row);
            let param = `?area=${this.area}&id=${row.id}&name=${row.name}`;
            axios.post('/model/download'+param
                ,{area:this.area,id:row.id, name:row.name},{responseType: 'blob'})
                .then(response =>{
                    let data = response.data;
                if (!data) {
                    return
                }
                console.log(response);
                let contentDisposition = response.headers['content-disposition'];
                let fileName = contentDisposition.substring(contentDisposition.indexOf('=')+1);
                console.log('fileName=' + fileName);
                let url = window.URL.createObjectURL(new Blob([data]))
                let a = document.createElement('a')
                a.style.display = 'none'
                a.href = url
                a.setAttribute('download',fileName)
                document.body.appendChild(a)
                //点击下载
                a.click()
                // 下载完成移除元素
                document.body.removeChild(a);
                // 释放掉blob对象
                window.URL.revokeObjectURL(url);
                console.log("下载完成");
                }).catch(function (error) {
                console.log(error);
            });
        }

说明:由于下载的是文件流,所以请求的方法配置参数需要把返回类型设置为blob,eg:responseType: 'blob',另外注意axios的get和post方法的传参顺序问题。get方法第一个参数是url+ur后面的参数,第二个参数是请求配置参数;post:有三个参数,第一个是url,第二个是请求参数,第三个是请求的配置数据。

下载下来的blob,我们需要创建一个url来指定这个流,创建html的a标签,设置下载属性download,下载的url,最后调用click方法进行下载,下载完成文件以后记得删除创建的a标签,释放资源。

 

后端核心代码:

@PostMapping("/download")
    @ResponseBody
    public ResponseResult download(HttpServletRequest request, HttpServletResponse response, @RequestBody ModelVO modelVO) throws UnsupportedEncodingException {
        //.......................业务代码省略,下面给出下载核心代码
        //下载逻辑:
        // 设置强制下载不打开
        response.setContentType("application/force-download");
        // 设置文件名,fileName是下载文件的文件名
        response.addHeader("Content-Disposition",
                "attachment;fileName=" + URLEncoder.encode(downloadModel.getName()+".pdf","UTF-8"));
        byte[] buffer = new byte[1024];
		//需要下载文件的字节数组
        byte[] modelByte = Base64Utils.base64String2Byte(downloadModel.getContent());
        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(modelByte);
        try {
            OutputStream out = response.getOutputStream();
            int i = byteArrayInputStream.read(buffer);
            while (i != -1) {
                out.write(buffer, 0, i);
                i = byteArrayInputStream.read(buffer);
            }
            out.flush();
            System.out.println("********************下载success**************");
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            if (byteArrayInputStream != null) {
                try {
                    byteArrayInputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return ResponseResult.success();
    }

说明:ModelVO是封装的一个类,用来接收post传输的参数,我的项目里使用了thymeleaf模板,所以下载接口需要添加@ResponseBody,不然以为不是接口,是需要返回一个页面的,下载接口是支持get和post,只是get和post方式的话,传参不一样,方法上面的注解也不一样,@PostMapping或@GetMapping