SpringBoot+ElementUI实现通用文件下载请求(全流程图文详细教程)

摘要:
实现既然是实现通用下载接口,就要实现在后端配置一个下载文件的路径,在前端进行下载请求时传递要下载的文件的名字,然后请求公共接口进行下载。引入方式import{download}from"@/utils/badao";在utils下的badao.js中//通用下载方法exportfunctiondownload{window.location.href=baseURL+"/common/download?fileName="+encodeURI+"&delete="+false;}将此方法进行暴露,作为公共方法。然后在这个url对应SprinBoot后台接口方法中@GetMappingpublicvoidfileDownload{try{if(!FileUtils.isValidFilename){thrownewException(StringUtils.format("文件名称({})非法,不允许下载。",fileName));}StringrealFileName=System.currentTimeMillis()+fileName.substring;StringfilePath=RuoYiConfig.getDownloadPath()+fileName;response.setCharacterEncoding;response.setContentType;response.setHeader;FileUtils.writeBytes;if{FileUtils.deleteFile;}}catch{log.error;}}首先调用了文件处理工具类的验证方法,验证文件名称是否合法。
场景

在某些场景下需要前端浏览器从服务器端下载文件,比如需要下载导入Excel的模板。

SpringBoot+ElementUI实现通用文件下载请求(全流程图文详细教程)第1张

注:

博客:
https://blog.csdn.net/badao_liumang_qizhi
关注公众号
霸道的程序猿
获取编程相关电子书、教程推送与免费下载。

实现

既然是实现通用下载接口,就要实现在后端配置一个下载文件的路径,在前端进行下载请求时传递要下载的文件的名字,然后请求公共接口进行下载。

首先是在前端使用ElementUI的el-link添加一个下载链接

          <el-link
            type="info"style="font-size:12px"@click="downloadTemplate('lxszTemplate.xlsx')"
          >下载模板</el-link>

这里设置了其点击事件是调用downloadTemplate方法并传递一个文件名参数,这个文件名就是要下载的文件名。

然后在对应的点击事件中

downloadTemplate(value) {
      download(value)
        .then((response) =>{})
        .catch((error) =>{
          alert("错误:" +error);
        });
    },

这里执行了一个download方法并传递文件名参数。

这个download方法是引用的第三方js中作为公共方法的。

引入方式

import { download } from "@/utils/badao";

在utils下的badao.js中

//通用下载方法
export function download(fileName) {
 window.location.href = baseURL + "/common/download?fileName=" + encodeURI(fileName) + "&delete=" + false;
}

将此方法进行暴露,作为公共方法。

在通用下载方法中使页面跳转window.location.href ,对应的url是SpringBoot中后台的接口。

这里的baseURL是在badao.js中声明的常量

const baseURL = process.env.VUE_APP_BASE_API

对此常量的赋值是取得全局变量process的属性,它对应的是在vue.config.js中配置的代理的地址

proxy: {
      [process.env.VUE_APP_BASE_API]: {
        target: `http://localhost:8080`,
        changeOrigin: true,
        pathRewrite: {
          ['^' + process.env.VUE_APP_BASE_API]: ''}
      }
    },

这里是我本地的8080端口。

然后在上面的通用的下载方法中在URL中还拼接了两个参数

一个是文件名参数,调用的js的encodeURI方法可以将字符串作为URL进行编码,一个是是否删除的参数,默认是false,作为下载成功后是否将文件给删除,即实现单次下载还是多次下载。

然后在这个url对应SprinBoot后台接口方法中

    @GetMapping("common/download")
    public voidfileDownload(String fileName, Boolean delete, HttpServletResponse response, HttpServletRequest request) {
        try{
            if (!FileUtils.isValidFilename(fileName)) {
                throw new Exception(StringUtils.format("文件名称({})非法,不允许下载。 ", fileName));
            }
            String realFileName = System.currentTimeMillis() + fileName.substring(fileName.indexOf("_") + 1);
            String filePath = RuoYiConfig.getDownloadPath() +fileName;

            response.setCharacterEncoding("utf-8");
            response.setContentType("multipart/form-data");
            response.setHeader("Content-Disposition",
                    "attachment;fileName=" +FileUtils.setFileDownloadHeader(request, realFileName));
            FileUtils.writeBytes(filePath, response.getOutputStream());
            if(delete) {
                FileUtils.deleteFile(filePath);
            }
        } catch(Exception e) {
            log.error("下载文件失败", e);
        }
    }

首先调用了文件处理工具类的验证方法,验证文件名称是否合法。

这里是设置了指定文件名称格式。

方法实现

    public staticboolean isValidFilename(String filename)
    {
        returnfilename.matches(FILENAME_PATTERN);
    }

其中参数为常量

public static String FILENAME_PATTERN = "[a-zA-Z0-9_\-\|\.\u4e00-\u9fa5]+";

下面是对服务器上文件路径的获取

String filePath = RuoYiConfig.getDownloadPath() + fileName;

其中RuoyiConfig是配置类,用来读取项目相关配置,即配置在application.yml中的内容。

import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

/**
 * 读取项目相关配置
 * 
 * @author ruoyi
 */@Component
@ConfigurationProperties(prefix = "ruoyi")
public classRuoYiConfig
{
    /** 项目名称 */
    privateString name;

    /** 版本 */
    privateString version;

    /** 版权年份 */
    privateString copyrightYear;

    /** 实例演示开关 */
    privateboolean demoEnabled;

    /** 上传路径 */
    private staticString profile;

    /** 获取地址开关 */
    private staticboolean addressEnabled;

    publicString getName()
    {
        returnname;
    }

    public voidsetName(String name)
    {
        this.name =name;
    }

    publicString getVersion()
    {
        returnversion;
    }

    public voidsetVersion(String version)
    {
        this.version =version;
    }

    publicString getCopyrightYear()
    {
        returncopyrightYear;
    }

    public voidsetCopyrightYear(String copyrightYear)
    {
        this.copyrightYear =copyrightYear;
    }

    publicboolean isDemoEnabled()
    {
        returndemoEnabled;
    }

    public voidsetDemoEnabled(boolean demoEnabled)
    {
        this.demoEnabled =demoEnabled;
    }

    public staticString getProfile()
    {
        returnprofile;
    }

    public voidsetProfile(String profile)
    {
        RuoYiConfig.profile =profile;
    }

    public staticboolean isAddressEnabled()
    {
        returnaddressEnabled;
    }

    public voidsetAddressEnabled(boolean addressEnabled)
    {
        RuoYiConfig.addressEnabled =addressEnabled;
    }

    /**
     * 获取头像上传路径
     */
    public staticString getAvatarPath()
    {
        return getProfile() + "/avatar";
    }

    /**
     * 获取下载路径
     */
    public staticString getDownloadPath()
    {
        return getProfile() + "/download/";
    }

    /**
     * 获取上传路径
     */
    public staticString getUploadPath()
    {
        return getProfile() + "/upload";
    }
}

这里的getDownloadPath就是获取设置的下载路径的方法,方法具体实现

    public staticString getDownloadPath()
    {
        return getProfile() + "/download/";
    }

就是返回profile这个节点

    public staticString getProfile()
    {
        returnprofile;
    }

因为使用了@ConfigurationProperties(prefix = "ruoyi")

所以在对应的application.yml中获取profile就是获取ruoyi下的profile的值

SpringBoot+ElementUI实现通用文件下载请求(全流程图文详细教程)第2张

这里配置的路径加上/download/再加上一个文件名就是服务器上要下载的模板文件的位置。

所以要提前将此文件放置在服务器上对应的位置。

这里服务器是我本地

SpringBoot+ElementUI实现通用文件下载请求(全流程图文详细教程)第3张

然后对文件名进行一个添加时间戳的操作,防止多次下载重名问题。

然后设置响应编码、响应头、响应类型。

然后调用了文件工具类的输出文件到Byte数组的方法writeBytes

方法实现

    /**
     * 输出指定文件的byte数组
     * 
     * @param filePath 文件路径
     * @param os 输出流
     * @return
     */
    public static voidwriteBytes(String filePath, OutputStream os) throws IOException
    {
        FileInputStream fis = null;
        try{
            File file = newFile(filePath);
            if (!file.exists())
            {
                throw newFileNotFoundException(filePath);
            }
            fis = newFileInputStream(file);
            byte[] b = new byte[1024];
            intlength;
            while ((length = fis.read(b)) > 0)
            {
                os.write(b, 0, length);
            }
        }
        catch(IOException e)
        {
            throwe;
        }
        finally{
            if (os != null)
            {
                try{
                    os.close();
                }
                catch(IOException e1)
                {
                    e1.printStackTrace();
                }
            }
            if (fis != null)
            {
                try{
                    fis.close();
                }
                catch(IOException e1)
                {
                    e1.printStackTrace();
                }
            }
        }
    }

以及下载文件名重新编码的方法setFileDownloadHeader

方法实现

    /**
     * 下载文件名重新编码
     * 
     * @param request 请求对象
     * @param fileName 文件名
     * @return 编码后的文件名
     */
    public staticString setFileDownloadHeader(HttpServletRequest request, String fileName)
            throws UnsupportedEncodingException
    {
        final String agent = request.getHeader("USER-AGENT");
        String filename =fileName;
        if (agent.contains("MSIE"))
        {
            //IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        }
        else if (agent.contains("Firefox"))
        {
            //火狐浏览器
            filename = new String(fileName.getBytes(), "ISO8859-1");
        }
        else if (agent.contains("Chrome"))
        {
            //google浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        else{
            //其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        returnfilename;
    }

然后根据传递的参数是否删除模板文件,执行删除的工具类方法deleteFile

方法实现

    /**
     * 删除文件
     * 
     * @param filePath 文件
     * @return
     */
    public staticboolean deleteFile(String filePath)
    {
        boolean flag = false;
        File file = newFile(filePath);
        //路径为文件且不为空则进行删除
        if (file.isFile() &&file.exists())
        {
            file.delete();
            flag = true;
        }
        returnflag;
    }

完整的文件操作工具类代码

package com.ruoyi.common.utils.file;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import javax.servlet.http.HttpServletRequest;

/**
 * 文件处理工具类
 * 
 * @author ruoyi
 */
public classFileUtils
{
    public static String FILENAME_PATTERN = "[a-zA-Z0-9_\-\|\.\u4e00-\u9fa5]+";

    /**
     * 输出指定文件的byte数组
     * 
     * @param filePath 文件路径
     * @param os 输出流
     * @return
     */
    public static voidwriteBytes(String filePath, OutputStream os) throws IOException
    {
        FileInputStream fis = null;
        try{
            File file = newFile(filePath);
            if (!file.exists())
            {
                throw newFileNotFoundException(filePath);
            }
            fis = newFileInputStream(file);
            byte[] b = new byte[1024];
            intlength;
            while ((length = fis.read(b)) > 0)
            {
                os.write(b, 0, length);
            }
        }
        catch(IOException e)
        {
            throwe;
        }
        finally{
            if (os != null)
            {
                try{
                    os.close();
                }
                catch(IOException e1)
                {
                    e1.printStackTrace();
                }
            }
            if (fis != null)
            {
                try{
                    fis.close();
                }
                catch(IOException e1)
                {
                    e1.printStackTrace();
                }
            }
        }
    }

    /**
     * 删除文件
     * 
     * @param filePath 文件
     * @return
     */
    public staticboolean deleteFile(String filePath)
    {
        boolean flag = false;
        File file = newFile(filePath);
        //路径为文件且不为空则进行删除
        if (file.isFile() &&file.exists())
        {
            file.delete();
            flag = true;
        }
        returnflag;
    }

    /**
     * 文件名称验证
     * 
     * @param filename 文件名称
     * @return true 正常 false 非法
     */
    public staticboolean isValidFilename(String filename)
    {
        returnfilename.matches(FILENAME_PATTERN);
    }

    /**
     * 下载文件名重新编码
     * 
     * @param request 请求对象
     * @param fileName 文件名
     * @return 编码后的文件名
     */
    public staticString setFileDownloadHeader(HttpServletRequest request, String fileName)
            throws UnsupportedEncodingException
    {
        final String agent = request.getHeader("USER-AGENT");
        String filename =fileName;
        if (agent.contains("MSIE"))
        {
            //IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        }
        else if (agent.contains("Firefox"))
        {
            //火狐浏览器
            filename = new String(fileName.getBytes(), "ISO8859-1");
        }
        else if (agent.contains("Chrome"))
        {
            //google浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        else{
            //其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        returnfilename;
    }
}

 

免责声明:文章转载自《SpringBoot+ElementUI实现通用文件下载请求(全流程图文详细教程)》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Chrome 浏览器快捷键改变 Windows 用户文件夹默认位置下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

asp.net时间显示

DateTime dt = DateTime.Now;//    Label1.Text = dt.ToString();//2005-11-5 13:21:25//    Label2.Text = dt.ToFileTime().ToString();//127756416859912816//    Label3.Text = dt.ToFileTi...

十九、多文件上传(ajaxFileupload实现多文件上传功能)

来源于https://www.jb51.net/article/128647.htm 打开google 搜索"ajaxFileupload' ‘多文件上传"可以搜到许许多多类似的,那我为什么还要写一下呢?一个是对之前大神的贡献表示感谢;二个是自己知识的总结;三个是自己在原有的基础上改动了下,在此记录,可能帮助其他朋友。 用过这个插件的都知道这个插件的基本用...

hw1 problem2

problem 2没什么好说的,用了String里的substring() 和concat()method。看地里有人说可以新建一个StringBuilder constructor,然后用里面的deleteCharAt()method  试了一下是可以的,还要再用一个toString() method  把类型转换回String. 在String上做什么...

解决Android与服务器交互大容量数据问题

对于目前的状况来说,移动终端的网络状况没有PC网络状况那么理想。在一个Android应用中,如果需要接收来自服务器的大容量数据,那么就不得不考虑客户的流量问题。本文根据笔者的一个项目实战经验出发,解决大容量数据的交互问题,解决数据大小会根据实际情况动态切换问题(服务器动态选择是否要压缩数据,客户端动态解析数据是否是被压缩的),还有数据交互的编码问题。 解决...

使用ganymed工具调用ssh2

需要引入ganymed-ssh2-build210.jar包。 其实很简单。所以直接贴代码,代码说话。 package com.eshore.framework.util; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream;...

你注意到 .Net Framework 和 .Net Core 中使用 Session 的区别了吗?

起因 在测试一个例子时发现的问题,这个示例实现的功能是刷新页面也能保持表格锁定列的状态,先看下页面的完成效果: 测试中发现,几乎相同的代码: 在 FineUIMvc(Net Framework)下没有问题:http://mvc.fineui.com/#/GridLockColumn/SaveToDB 但是在 FineUICore(Net Core)下...