SpringMVC跨服务器文件上传

  上一次讲解了SpringMVC对于传统文件的上传,SpringMVC主要完成了对request请求的解析。这一节来讲解SpringMVC如何跨服务器进行文件上传

  为什么需要跨服务器进行文件上传?原因为了使我们服务器能专一并且高效的做事情,可以把服务器分为应用服务器,缓存服务器,数据库服务器,文件服务器。而文件服务器就是专门用来存储应用上传的文件,当客户端请求时,直接从文件服务器来进行响应,而上传文件时,通过应用服务器将文件传到文件服务器,也即是下图的逻辑:
在这里插入图片描述

演示需要两个服务器,我们这里新建一个项目即可,同时在本机上启动两台服务器,只不过端口不一样。
在这里插入图片描述
两个值都要进行修改,创建好这个项目后,我们就启动Tomcat,专门用来接收文件

  1. 文件放在我们自己创建的一个文件夹中,取名为uploads,这里Tomcat选择war_explode开发模式,而不是war,主要原因是war是需要部署到Tomcat下的webapp下,war_explode模式可以直接以文件夹方式部署,这样我们更好确认路径,直接在target文件夹下创建一个uploads的文件夹,如果没有创建,后面有可能会有409的错误
  2. 我们需要在web.xml中配置写的请求,因为这是一个put类型的请求,所以我们需要支持写入操作,这里直接替换所有web.xml的内容。如果没有更改,可能会有403的错误
<!DOCTYPE web-app PUBLIC
        "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
        "http://java.sun.com/dtd/web-app_2_3.dtd" >

<!--suppress ALL -->
<web-app>
  <display-name>Archetype Created Web Application</display-name>

  <servlet>
    <servlet-name>default</servlet-name>
    <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class>
    <init-param>
      <param-name>debug</param-name>
      <param-value>0</param-value>
    </init-param>
    //主要是配置这里,只读为false
    <init-param>
      <param-name>readonly</param-name>
      <param-value>false</param-value>
    </init-param>
    <init-param>
      <param-name>listings</param-name>


      <param-value>false</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
  </servlet>
</web-app>

在创建好了文件服务器后,下面来看一下应用服务器这里如何编写,这里希望不懂SpringMVC原理的可以去阅读一下我的上一篇文章

前端页面form表单:

    <form action="/mvc/file/testRemoteFile" enctype="multipart/form-data" method="post">
        选择文件:<input type="file" name="upload"><br/>
        <input type="submit" value="上传">
    </form>

注意form表单的类型和提交方法的类型

提交该表单后,首先会经过文件解析器,这里给出文件解析器的配置:

    <!--配置文件解析器-->
    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <property name="maxUploadSize">
            <value>5242880</value>
        </property>
    </bean>

通过文件解析器,会将文件项以参数绑定的形式,发送到指定路由,看一下映射方法:

    @RequestMapping("/testRemoteFile")
    public String testRomoteFile(MultipartFile upload) throws Exception {
        System.out.println("testRomoteFile");
    }

从该方法中,我们可以通过文件项拿到二进制流和文件名称,但是我们要把该文件写入到文件服务器,而不是这台服务器,所以我们需要通过外部提供的jar包,打包成json,发送到文件服务器,先引入maven配置:

    <!--跨服务器json-->
    <dependency>
      <groupId>com.sun.jersey</groupId>
      <artifactId>jersey-core</artifactId>
      <version>1.18.1</version>
    </dependency>
    <dependency>
      <groupId>com.sun.jersey</groupId>
      <artifactId>jersey-client</artifactId>
      <version>1.18.1</version>
    </dependency>

引入配置后,就可以完后代码逻辑了,逻辑如下:

    @RequestMapping("/testRemoteFile")
    public String testRomoteFile(MultipartFile upload) throws Exception {
        System.out.println("testRomoteFile");
        //使用fileupload组件完成文件上传
        //上传的位置
        String path = "http://localhost:8080/file_server/uploads/";

        String fileName = upload.getOriginalFilename();
        String uuid = UUID.randomUUID().toString().replace("-", "");
        fileName = uuid + "_" + fileName;

        //创建客户端对象
        Client client = Client.create();

        //和图片服务器连接
        WebResource resource = client.resource(path + fileName);

        //上传文件
        resource.put(upload.getBytes());

        return "success";
    }

可以看到,我们通过创建一个客户端,把文件发送到文件服务器的指定位置,其实这个提交就是put请求,可以自己去了解一下8中不同的请求方式。

最后需要说的一个地方就是高版本的Tomcat会由于发送json格式的串中的特殊字符“{”,"}",报一个下面的错误:

Invalid character found in the request target. The valid characters are defined in RFC 7230 and RFC 问题

该错误的意思就是请求路径中有非法字符,是不允许的,我们可以更换低版本的Tomcat,这是最直接的办法,但是有可能没有安装低版本,我们可以配置Tocmat的配置文件

配置tomcat的catalina.properties,在最后一行添加如下内容
  tomcat.util.http.parser.HttpParser.requestTargetAllow=|{}

添加完成后,有可能仍会报错,这是因为"{"字符被允许了,但是有可能你的文件名中有中文,这里是为了学习跨服务器文件上传,所以可以传一个英文名称的文件

本篇文章查阅了如下博客,非常感谢:
https://www.cnblogs.com/lr393993507/p/7755867.html
https://www.cnblogs.com/STK0210/p/11216404.html

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值