• Spring MVC 中上传文件


    SpringMVC 中对文件上传做了封装,我们可以更加方便的实现文件上传。从 Spring3.1 开始,对于文件上传,提供了两个处理器:

    • CommonsMultipartResolver
    • StandardServletMultipartResolver·

    第一个处理器兼容性较好,可以兼容 Servlet3.0 之前的版本,但是它依赖了 commons-fileupload 这个第三方工具,所以如果使用这个,一定要添加 commons-fileupload 依赖。

    第二个处理器兼容性较差,它适用于 Servlet3.0 之后的版本,它不依赖第三方工具,使用它,可以直接做文件上传。

    CommonsMultipartResolver


    使用 CommonsMultipartResolver 做文件上传,需要首先添加 commons-fileupload 依赖,如下:

    <dependency>
        <groupId>commons-fileupload</groupId>
        <artifactId>commons-fileupload</artifactId>
        <version>1.4</version>
    </dependency>
    

    然后,在 SpringMVC 的配置文件中,配置 MultipartResolver

    <bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver" id="multipartResolver"/>
    

    注意,这个 Bean 一定要有 id,并且 id 必须是 multipartResolver

    接下来,创建 jsp 页面:

    <form action="/upload" method="post" enctype="multipart/form-data">
        <input type="file" name="file">
        <input type="submit" value="上传">
    </form>
    

    注意文件上传请求是 POST 请求,enctype 一定是 multipart/form-data,然后,开发文件上传接口:

    @Controller
    public class FileUploadController {
        
        SimpleDateFormat sdf = new SimpleDateFormat("/yyyy/MM/dd/");
    
        @RequestMapping("/upload")
        @ResponseBody
        public String upload(MultipartFile file, HttpServletRequest req) {
            
            String format = sdf.format(new Date());
            String realPath = req.getServletContext().getRealPath("/img") + format;
            File folder = new File(realPath);
            if (!folder.exists()) {
                folder.mkdirs();
            }
            String oldName = file.getOriginalFilename();
            String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf("."));
            try {
                file.transferTo(new File(folder, newName));
                String url = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName;
                return url;
            } catch (IOException e) {
                e.printStackTrace();
            }
            return "failed";
        }
    }
    

    这个文件上传方法中,一共做了四件事:

    1. 解决文件保存路径,这里是保存在项目运行目录下的 img 目录下,然后利用日期继续宁分类
    2. 处理文件名问题,使用 UUID 做新的文件名,用来代替旧的文件名,可以有效防止文件名冲突
    3. 保存文件
    4. 生成文件访问路径

    这里还有一个小问题,在 SpringMVC 中,静态资源默认都是被自动拦截的,无法访问,意味着上传成功的图片无法访问,因此,还需要我们在 SpringMVC 的配置文件中,再添加如下配置:

    <mvc:resources mapping="/**" location="/"/>
    

    完成之后,就可以访问 jsp 页面,做文件上传了。当然,默认的配置不一定满足我们的需求,我们还可以自己手动配置文件上传大小等:

    <bean class="org.springframework.web.multipart.commons.CommonsMultipartResolver" id="multipartResolver">
        <!--默认的编码-->
        <property name="defaultEncoding" value="UTF-8"/>
        <!--上传的总文件大小-->
        <property name="maxUploadSize" value="1048576"/>
        <!--上传的单个文件大小-->
        <property name="maxUploadSizePerFile" value="1048576"/>
        <!--内存中最大的数据量,超过这个数据量,数据就要开始往硬盘中写了-->
        <property name="maxInMemorySize" value="4096"/>
        <!--临时目录,超过 maxInMemorySize 配置的大小后,数据开始往临时目录写,等全部上传完成后,再将数据合并到正式的文件上传目录-->
        <property name="uploadTempDir" value="file:///E:\tmp"/>
    </bean>
    

    StandardServletMultipartResolver


    这种文件上传方式,不需要依赖第三方 jar(主要是不需要添加 commons-fileupload 这个依赖),但是也不支持 Servlet3.0 之前的版本。

    使用 StandardServletMultipartResolver,那我们首先在 SpringMVC 的配置文件中,配置这个 Bean:

    <bean class="org.springframework.web.multipart.support.StandardServletMultipartResolver" id="multipartResolver">
    </bean>
    

    注意,这里 Bean 的名字依然叫 multipartResolver

    配置完成后,注意,这个 Bean 无法直接配置上传文件大小等限制。需要在 web.xml 中进行配置(这里,即使不需要限制文件上传大小,也需要在 web.xml 中配置 multipart-config):

    <servlet>
        <servlet-name>springmvc</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:spring-servlet.xml</param-value>
        </init-param>
        <multipart-config>
            <!--文件保存的临时目录,这个目录系统不会主动创建-->
            <location>E:\temp</location>
            <!--上传的单个文件大小-->
            <max-file-size>1048576</max-file-size>
            <!--上传的总文件大小-->
            <max-request-size>1048576</max-request-size>
            <!--这个就是内存中保存的文件最大大小-->
            <file-size-threshold>4096</file-size-threshold>
        </multipart-config>
    </servlet>
    <servlet-mapping>
        <servlet-name>springmvc</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>
    

    配置完成后,就可以测试文件上传了,测试方式和上面一样。

    多文件上传


    多文件上传分为两种,一种是 key 相同的文件,另一种是 key 不同的文件。

    key 相同的文件

    这种上传,前端页面一般如下:

    <form action="/upload2" method="post" enctype="multipart/form-data">
        <input type="file" name="files" multiple>
        <input type="submit" value="上传">
    </form>
    

    主要是 input 节点中多了 multiple 属性。后端用一个数组来接收文件即可:

    @RequestMapping("/upload2")
    @ResponseBody
    public void upload2(MultipartFile[] files, HttpServletRequest req) {
        
        String format = sdf.format(new Date());
        String realPath = req.getServletContext().getRealPath("/img") + format;
        File folder = new File(realPath);
        if (!folder.exists()) {
            folder.mkdirs();
        }
        try {
            for (MultipartFile file : files) {
                String oldName = file.getOriginalFilename();
                String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf("."));
                file.transferTo(new File(folder, newName));
                String url = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName;
                System.out.println(url);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    

    key 不同的文件

    key 不同的,一般前端定义如下:

    <form action="/upload3" method="post" enctype="multipart/form-data">
        <input type="file" name="file1">
        <input type="file" name="file2">
        <input type="submit" value="上传">
    </form>
    

    这种,在后端用不同的变量来接收就行了:

    @RequestMapping("/upload3")
    @ResponseBody
    public void upload3(MultipartFile file1, MultipartFile file2, HttpServletRequest req) {
        
        String format = sdf.format(new Date());
        String realPath = req.getServletContext().getRealPath("/img") + format;
        File folder = new File(realPath);
        if (!folder.exists()) {
            folder.mkdirs();
        }
        try {
            String oldName = file1.getOriginalFilename();
            String newName = UUID.randomUUID().toString() + oldName.substring(oldName.lastIndexOf("."));
            file1.transferTo(new File(folder, newName));
            String url1 = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName;
            System.out.println(url1);
            String oldName2 = file2.getOriginalFilename();
            String newName2 = UUID.randomUUID().toString() + oldName2.substring(oldName2.lastIndexOf("."));
            file2.transferTo(new File(folder, newName2));
            String url2 = req.getScheme() + "://" + req.getServerName() + ":" + req.getServerPort() + "/img" + format + newName2;
            System.out.println(url2);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    

    全局异常处理


    项目中,可能会抛出多个异常,我们不可以直接将异常的堆栈信息展示给用户,有两个原因:

    1. 用户体验不好
    2. 非常不安全

    所以,针对异常,我们可以自定义异常处理,SpringMVC 中,针对全局异常也提供了相应的解决方案,主要是通过 @ControllerAdvice@ExceptionHandler 两个注解来处理的。

    以文件上传大小超出限制为例,自定义异常,只需要提供一个异常处理类即可,通过指定 @ExceptionHandler 属性值 MaxUploadSizeExceededException.class 只会拦截文件上传异常,其他异常和它没关系:

    @ControllerAdvice  //表示这是一个增强版的 Controller,主要用来做全局数据处理
    public class MyException {
        
        @ExceptionHandler(MaxUploadSizeExceededException.class)
        public ModelAndView fileuploadException(Exception e) {
            ModelAndView error = new ModelAndView("error");
            error.addObject("error", e.getMessage());
            return error;
        }
    }
    

    在这里:

    • @ControllerAdvice 表示这是一个增强版的 Controller,主要用来做全局数据处理
    • @ExceptionHandler 表示这是一个异常处理方法,这个注解的参数,表示需要拦截的异常,参数为 Exception 表示拦截所有异常,这里也可以具体到某一个异常,如果具体到某一个异常,那么发生了其他异常则不会被拦截到。
    • 异常方法的定义,和 Controller 中方法的定义一样,可以返回 ModelAndview,也可以返回 String 或者 void

    更多干货请移步:https://antoniopeng.com

  • 相关阅读:
    Laravel Vuejs 实战:开发知乎 (21)前后端分离 API token 认证
    Laravel Vuejs 实战:开发知乎 (20)使用 Vuejs 组件化
    Laravel Vuejs 实战:开发知乎 (18-19)用户关注问题
    Laravel Vuejs 实战:开发知乎 (17)实现提交答案
    Laravel Vuejs 实战:开发知乎 (16)创建问题的答案 Answer
    Laravel Vuejs 实战:开发知乎 (15)问题 Feed 和删除问题
    Laravel Vuejs 实战:开发知乎 (13)实现编辑问题
    Laravel Vuejs 实战:开发知乎 (12)使用 Repository 模式
    Laravel Vuejs 实战:开发知乎 (11)实现选择话题整个流程
    Jmeter函数作用域实时取值覆盖[针对HTTP Request等控制器]
  • 原文地址:https://www.cnblogs.com/antoniopeng/p/14371581.html
Copyright © 2020-2023  润新知