SpringMvc:执行原理+RestFul+JSON交互+Ajax+拦截器+文件上传下载

在这里插入图片描述

SpringMvc

官网地址
SpringMVC官网地址
参考博客

SpringMVC是Spring Framework的一部分,是基于Java实现MVC的轻量级架构

  • 模型(Model):
    • 业务逻辑
    • 保存数据的状态
  • 视图(View)
    • 显示页面
  • 控制器(Controller)
    • 获取表单数据
    • 调用业务逻辑
    • 转向指定的页面

2345

SpringMVC执行原理

  • 中心控制器(DispatcherServlet)

Spring的web框架围绕DispatcherServlet设计,将请求分发到不同的处理器
以请求为驱动,围绕一个中心Servlet分派请求及提供其他功能,
DispatcherServlet是一个实际的Servlet(继承HttpServlet基类)

  • 请求执行流程图
    1345
  1. DispatcherServlet表示前端控制器,是整个SpringMVC的核心,拦截用户发出的请求
  2. HandlerMapping为处理器映射。DispatcherServlet调用,HanderMapping根据请求查找Handler
  3. HandlerExecetion表示具体的Handler,其主要作用是根据url查找控制器
  4. HandlerExecetion将解析后的信息传递给DispatcherServlet
  5. HandlerAdapter表示处理适配器,按照特定规则去执行Handler
  6. Handler让具体的Controller执行
  7. Controller将具体的执行信息返回给HandlerAdapter,如ModelAndView
  8. HandlerAdapter将视图逻辑名或模型传递给DispaticherServlet
  9. DispaticherServlet调用视图解析器(ViewResolver)来解析HandlerAdapter传递的逻辑视图名
  10. 视图解析器将解析的逻辑视图名传给DispatcherServlet
  11. DispaticherServlet根据视图解析器解析视图结果,调用具体的视图
  12. 最终视图呈现给用户
  • springMVC必须配置的三大件:处理器映射器、处理器适配器、视图适配器

SpringMVC基本项目搭建(不使用注解)

  1. 在web.xml中配置DispatcherServlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">

    <!--配置DispatcherServlet(请求分发器,前端控制器):springMVC的核心-->
    <servlet>
        <servlet-name>springmvc</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <!--DispatcherServlet要绑定Spring配置文件-->
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:springmvc-servlet.xml</param-value>
        </init-param>
        <!--设置启动级别:1-->
        <load-on-startup>1</load-on-startup>
    </servlet>
    <!--spring MVC中/和/*
        /:只匹配所有请求,不会去匹配jsp页面
        /*:匹配所有请求和jsp页面
        -->
    <servlet-mapping>
        <servlet-name>springmvc</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>


<!--    <welcome-file-list>-->
<!--        <welcome-file>index.jsp</welcome-file>-->
<!--    </welcome-file-list>-->
</web-app>
  1. 配置spring的springmvc-servlet.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/tx
        http://www.springframework.org/schema/tx/spring-tx.xsd">
    <!--处理器映射器-->
    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
    <!--处理器适配器-->
    <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
    <!--视图解析器-->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
        <!--前缀-->
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <!--后缀-->
        <property name="suffix" value=".jsp"/>
    </bean>

    <!--BeanNameUrlHandlerMapping:根据bean的名字查找-->
    <bean id="/hello" class="com.huang.controller.HelloController"/>

</beans>
  1. 编写Controller接口实现类
public class HelloController implements Controller {

    public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        ModelAndView modelAndView = new ModelAndView();
        System.out.println("==========================================");
        //业务代码
        String result="hello springMVC";
        modelAndView.addObject("msg",result);
        //视图跳转
        modelAndView.setViewName("test");
        return modelAndView;
    }
}

SpringMVC基本项目搭建(使用注解)

  1. 在web.xml中配置DispatcherServlet:和不使用注解是一样的
  2. 配置spring的springmvc-servlet.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/mvc
        https://www.springframework.org/schema/mvc/spring-mvc.xsd">

<!--使用注解配置方法-->
    <!--自动扫描包,让指定包下的的注解生效,由IOC统一管理-->
    <context:component-scan base-package="com.huang.controller"/>
    <!--让springMVC不处理静态资源-->
    <mvc:default-servlet-handler/>
    <!--支持MVC注解驱动:
            在spring中一般采用@RequestMapping注解完成映射关系
            annotation-driven配置帮助使这个注解生效,-->
    <mvc:annotation-driven/>

    <!--视图解析器-->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
        <!--前缀-->
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <!--后缀-->
        <property name="suffix" value=".jsp"/>
    </bean>
</beans>
  1. 编写controller实现方法:
@Controller
public class helloController2 {
    @RequestMapping("/hello")
    public String hello(Model model){
        //封装数据
        model.addAttribute("msg","hello controller 2");
        return "test";//会被视图解析器处理
    }
}
Controller配置

Controller(控制器)

  • 负责解析用户请求并将其转换为一个模型
  • 一个控制器可以包含多个方法

实现Controller的两种方式

  • 继承controller接口,注册bean
    • 一个控制器种只有一个方法,如果有多个方法,就要定义多个类
  • 使用注解:@Controller,@RequestMapping("")
    • 类被Spring接管,如果返回值使string并且有具体的页面可以跳转,就会被视图解析器解析
RequestMapping
  • 用于映射url到控制器类或一个特定的处理程序的方法
  • 可用于类或方法上
    • 用于类上,表示类种的所有响应请求的方法都是以该地址作为父路径

RestFul风格

  • Restful就是一个资源定位及资源操作的风格
  • 可以更简洁,更有层次,更易实现缓存机制
  • 传统的方式操作资源:通过不同的参数实现不同的效果
  • 使用Restful操作资源:通过不同的请求方式来实现不同的效果
  • 简洁安全高效
    //映射访问路径,不指定提交方式
    @RequestMapping("/commit/{p1}/{p2}")
    public String index(@PathVariable int p1, @PathVariable int p2, Model model){
        int result = p1+p2;
        //Spring MVC会自动实例化一个Model对象用于向视图中传值
        model.addAttribute("msg", "结果:"+result);
        //返回视图位置
        return "test";
    }
    //映射访问路径,必须是GET请求
    @RequestMapping(value = "/commit/{p1}/{p2}",method = {RequestMethod.GET})
    public String index2(@PathVariable int p1, @PathVariable int p2, Model model){
        int result = p1+p2;
        model.addAttribute("msg", "结果:"+result);
        return "test";
    }
    //映射访问路径,必须是GET请求的另一种实现方式
    @GetMapping("/commit/{p1}/{p2}")
    public String index3(@PathVariable int p1, @PathVariable int p2, Model model){
        int result = p1+p2;
        model.addAttribute("msg", "结果:"+result);
        return "test";
    }

结果跳转

通过视图解析器跳转

  • 设置ModelAndView对象,根据view的名称,和视图解析器跳转到指定的页面
  • 页面:(视图解析器前缀)+viewName+(视图解析器后缀)
 @GetMapping("/commit/{p1}/{p2}")
    public String index3(@PathVariable int p1, @PathVariable int p2, Model model){
        int result = p1+p2;
        model.addAttribute("msg", "结果:"+result);
        return "test";
    }

通过SpringMVC来实现转发和重定向

  • 视图解析器主要来拼接路径

无需视图解析器

        @RequestMapping("/t1")
        public String test1(){
            //转发
            return "/WEB-INF/jsp/test.jsp";
        }
        @RequestMapping("/t2")
        public String test2(){
            //转发二
            return "forward:/WEB-INF/jsp/test.jsp";
        }

        @RequestMapping("/t3")
        public String test3(){
            //重定向
            return "redirect:/WEB-INF/jsp/test.jsp";
        }

有视图解析器的:不需要使用全路径

接收参数及数据显示

接收参数

//http://localhost:8080/user/t1?username=huang
    @GetMapping("/user/t1")
    public String test01(@RequestParam("username") String name, Model model){
        //接收前端参数
        System.out.println(name);
        //将结果返回前端
        model.addAttribute("msg","信息"+name);
        //跳转视图
        return "test";
    }

接收对象

   //接收的是一个对象:id name age
    // http://localhost:8080/user/t2?name=huang&id=1&age=15
    @GetMapping("/user/t2")
    public String test02(User user, Model model){
        //接收前端参数
        //接收前端的参数,如果参数名和方法参数名一致,则匹配
        //如果方法参数是一个对象,则匹配对象字段名,不一致返回null
        System.out.println(""+user.getId()+user.getName()+user.getAge());
        //将结果返回前端
        model.addAttribute("msg","信息"+user.getId()+user.getName()+user.getAge());
        //跳转视图
        return "test";
    }

返回方式

  1. ModelAndView:继承Controller 接口,实现方法
  2. ModelMap:和Model的使用方法差不多
  3. Model
  • Model 只有寥寥几个方法只适合用于储存数据,简化了新手对于Model对象的操作和理解;
  • ModelMap 继承了 LinkedMap ,除了实现了自身的一些方法,同样的继承 LinkedMap 的方法和特性;
  • ModelAndView 可以在储存数据的同时,可以进行设置返回的逻辑视图,进行控制展示层的跳转。

乱码问题

使用SpringMVC提供的过滤器

  • 修改Web.xml
  • 能解决大部分的乱码问题
    <!--配置SpringMVC的乱码过滤器-->
    <filter>
        <filter-name>encoding</filter-name>
        <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
        <init-param>
            <param-name>encoding</param-name>
            <param-value>utf-8</param-value>
        </init-param>
    </filter>
    <filter-mapping>
        <filter-name>encoding</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

自定义过滤器

  • 终极解决方案
  • 编写类GenericEncodingFilter
  • 注册过滤器
    <filter>
        <filter-name>encoding2</filter-name>
        <filter-class>com.huang.Filter.GenericEncodingFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>encoding2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
package com.huang.Filter;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Map;
/**
 * 解决get和post请求 全部乱码的过滤器
 */
public class GenericEncodingFilter implements Filter {

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        //处理response的字符编码
        HttpServletResponse myResponse=(HttpServletResponse) response;
        myResponse.setContentType("text/html;charset=UTF-8");

        // 转型为与协议相关对象
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        // 对request包装增强
        HttpServletRequest myrequest = new MyRequest(httpServletRequest);
        chain.doFilter(myrequest, response);
    }

    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }

}

//自定义request对象,HttpServletRequest的包装类
class MyRequest extends HttpServletRequestWrapper {

    private HttpServletRequest request;
    //是否编码的标记
    private boolean hasEncode;
    //定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
    public MyRequest(HttpServletRequest request) {
        super(request);// super必须写
        this.request = request;
    }

    // 对需要增强方法 进行覆盖
    @Override
    public Map getParameterMap() {
        // 先获得请求方式
        String method = request.getMethod();
        if (method.equalsIgnoreCase("post")) {
            // post请求
            try {
                // 处理post乱码
                request.setCharacterEncoding("utf-8");
                return request.getParameterMap();
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            }
        } else if (method.equalsIgnoreCase("get")) {
            // get请求
            Map<String, String[]> parameterMap = request.getParameterMap();
            if (!hasEncode) { // 确保get手动编码逻辑只运行一次
                for (String parameterName : parameterMap.keySet()) {
                    String[] values = parameterMap.get(parameterName);
                    if (values != null) {
                        for (int i = 0; i < values.length; i++) {
                            try {
                                // 处理get乱码
                                values[i] = new String(values[i]
                                        .getBytes("ISO-8859-1"), "utf-8");
                            } catch (UnsupportedEncodingException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
                hasEncode = true;
            }
            return parameterMap;
        }
        return super.getParameterMap();
    }

    //取一个值
    @Override
    public String getParameter(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        if (values == null) {
            return null;
        }
        return values[0]; // 取回参数的第一个值
    }

    //取所有值
    @Override
    public String[] getParameterValues(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        return values;
    }
}

JSON

  • JSON(JavaScript Object Notation, JS 对象标记) 是一种轻量级的数据交换格式,目前使用特别广泛。
  • 采用完全独立于编程语言的文本格式来存储和表示数据。
  • 简洁和清晰的层次结构使得 JSON 成为理想的数据交换语
  • 易于人阅读和编写,同时也易于机器解析和生成,并有效地提升网络传输效率。
  • 语法格式:
    • 对象表示为键值对,数据由逗号分隔
    • 花括号保存对象
    • 方括号保存数组

Jackson

  • 导入jar包:
        <!--jackson-databind -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.11.1</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-annotations -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-annotations</artifactId>
            <version>2.11.1</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-core -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.11.1</version>
        </dependency>
  • Json乱码统一解决:springmvc-servlet.xml中配置
<!--Json乱码解决-->
    <mvc:annotation-driven>
        <mvc:message-converters register-defaults="true">
            <bean class="org.springframework.http.converter.StringHttpMessageConverter">
                <constructor-arg value="UTF-8"/>
            </bean>
            <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
                <property name="objectMapper">
                    <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
                        <property name="failOnEmptyBeans" value="false"/>
                    </bean>
                </property>
            </bean>
        </mvc:message-converters>
    </mvc:annotation-driven>
  • 测试
//@Controller
@RestController   //加了这个注解不会走视图解析器,直接返回一个字符串
public class UserController{
	     //produces:指定响应体返回类型和编码
	    @RequestMapping("/user/t4")
		//@ResponseBody   //加了这个注解后就不会走视图解析器,会直接返回一个字符串
	    public String test04() throws JsonProcessingException {
	        ObjectMapper mapper=new ObjectMapper();
	        User user=new User(1,"黄",18);
	        String s = mapper.writeValueAsString(user);
	        return s;
	    }
}

FastJson

  • 导包
<dependency>
   <groupId>com.alibaba</groupId>
   <artifactId>fastjson</artifactId>
   <version>1.2.60</version>
</dependency>
  • 使用
    @RequestMapping("/user/t5")
    public String test05() throws JsonProcessingException {
        User user=new User(1,"黄",18);
        return JSON.toJSONString(user);
    }

Ajax

  • AJAX (Asynchronous JavaScript and XML):异步的 JavaScript 和 XML。
  • AJAX 是一种在无需重新加载整个网页的情况下,能够更新部分网页的技术

jQuery.ajax

  • Ajax的核心是XMLHttpRequest对象(XHR)。XHR为向服务器发送请求和解析服务器响应提供了接口。
  • 能够以异步方式从服务器获取新数据。
  • 通过 jQuery AJAX 方法,您能够使用 HTTP Get 和 HTTP Post 从远程服务器上请求文本、HTML、XML 或 JSON
  • jQuery Ajax本质就是 XMLHttpRequest
jQuery.ajax(...)
      部分参数:
            url:请求地址
            type:请求方式,GETPOST1.9.0之后用method)
        headers:请求头
            data:要发送的数据
    contentType:即将发送信息至服务器的内容编码类型(默认: "application/x-www-form-urlencoded; charset=UTF-8")
          async:是否异步
        timeout:设置请求超时时间(毫秒)
      beforeSend:发送请求前执行的函数(全局)
        complete:完成之后执行的回调函数(全局)
        success:成功之后执行的回调函数(全局)
          error:失败之后执行的回调函数(全局)
        accepts:通过请求头发送给服务器,告诉服务器当前客户端可接受的数据类型
        dataType:将服务器端返回的数据转换成指定类型
          "xml": 将服务器端返回的内容转换成xml格式
          "text": 将服务器端返回的内容转换成普通文本格式
          "html": 将服务器端返回的内容转换成普通文本格式,在插入DOM中时,如果包含JavaScript标签,则会尝试去执行。
        "script": 尝试将返回值当作JavaScript去执行,然后再将服务器端返回的内容转换成普通文本格式
          "json": 将服务器端返回的内容转换成相应的JavaScript对象
        "jsonp": JSONP 格式使用 JSONP 形式调用函数时,如 "myurl?callback=?" jQuery 将自动替换 ? 为正确的函数名,以执行回调函数

测试

  1. 配置静态资源过滤
<mvc:default-servlet-handler />
  1. 导入JQuery
    <script src="${pageContext.request.contextPath}/static/js/jquery-3.5.1.js"></script>
  1. 编写一个AjaxController
    @RequestMapping("/t1")
    public void testAjax1(String name, HttpServletResponse resource) throws IOException {
        System.out.println("-------------------t1:"+name);
        if(name.equals("huang")){
            resource.getWriter().write("success");
        }else {
            resource.getWriter().write("false");
        }
    }
    @RequestMapping("/t2")
    public List<Books> testAjax2(){
        List<Books> books=new ArrayList<Books>();
        books.add(new Books(1,"bookName",1,"aaa"));
        books.add(new Books(2,"bookName",1,"bbb"));
        books.add(new Books(3,"bookName",1,"ccc"));
        return books;
    }
  1. 编写index.jsp测试
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <head>
    <title>书籍</title>
    <script src="${pageContext.request.contextPath}/static/js/jquery-3.5.1.js"></script>
    <script>
      function a(){
        $.post({
          url:"${pageContext.request.contextPath}/t2",
          data:{"name":$("#username").val()},
          success:function (data) {
            alert(data);
          }
        })
      }
      
      //点击按钮
      $(function() {
        $("#btn").click(function () {
          console.log("hhhh")
          $.post({
            url:"${pageContext.request.contextPath}/t2",
            success:function (data) {
            //显示到页面
              let html="";
              for (let i = 0; i < data.length; i++) {
                html+="<tr>"
                        +"<td>"+data[i].bookID+"<td>"
                        +"<td>"+data[i].bookName+"<td>"
                        +"<td>"+data[i].bookCounts+"<td>"
                        +"<td>"+data[i].detail+"<td>"
                        +"</tr>"
              }
              $("#content").html(html)
            }
          })
        })
      });
    </script>
  </head>
  <body>

  <div>
    <%--失去焦点返送请求到后台--%>
    <p>用户名:<input type="text" id="username" onblur="a()"></p>
  </div>
  <div>
    <input type="button" id="btn" value="加载书籍">
    <table>
      <tr>
        <td>id</td>
        <td>name</td>
        <td>数量</td>
        <td>描述</td>
      </tr>
      <tbody id="content">
      </tbody>
    </table>
  </div>
  </body>
</html>

拦截器

  • SpringMVC的处理器拦截器类似于Servlet开发中的过滤器Filter
  • 用于对处理器进行预处理和后处理。
  • 过滤器与拦截器的区别:拦截器是AOP思想的具体应用。
  • 过滤器
    • servlet规范中的一部分,任何java web工程都可以使用
    • 在url-pattern中配置了/*之后,可以对所有要访问的资源进行拦截
  • 拦截器
    • 拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用
    • 拦截器只会拦截访问的控制器方法, 如果访问的是jsp/html/css/image/js是不会进行拦截的

自定义过滤器

  • 编写一个过滤器继承HandlerInterceptor 接口
public class myInterceptor implements HandlerInterceptor {
    //return true:执行下一个拦截器
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=============处理前=============");
        return false;
    }
    //可以写一些日志
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("=============处理后=============");
    }
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("==============清理==============");
    }
}
  • 在spring-mvc的配置文件中添加拦截器
<!--拦截器配置-->
    <mvc:interceptors>
        <mvc:interceptor>
            <!--/**代表这个请求下所有的请求-->
            <mvc:mapping path="/**"/>
            <bean class="com.huang.config.myInterceptor"/>
        </mvc:interceptor>
    </mvc:interceptors>

文件的上传

设置表单提交

  • 使用Spring的文件上传功能,则需要在上下文中配置MultipartResolver。
  • 为了能上传文件,必须将表单的method设置为POST,并将enctype设置为multipart/form-data。只有在这样的情况下,浏览器才会把用户选择的文件以二进制数据发送给服务器;
  • enctype 属性:
    • 默认方式(application/x-www=form-urlencoded):会将表单处理成URL编码方式
    • 以二进制流的方式(multipart/form-data):把文件域指定文件的内容也封装到请求参数中,不会对字符编码。
    • (text/plain):除了把空格转换为 “+” 号外,其他字符都不做编码处理,这种方式适用直接通过表单发送邮件。
<form action="${pageContext.request.contextPath}/insert" enctype="multipart/form-data" method="post">
   <input type="file" name="file"/>
   <input type="submit">
</form>

配置环境

  • SpringMVC的文件上传还需要依赖Apache Commons FileUpload的组件。
  • 当如配置依赖
<!--文件上传-->
<dependency>
   <groupId>commons-fileupload</groupId>
   <artifactId>commons-fileupload</artifactId>
   <version>1.3.3</version>
</dependency>
<!--servlet-api导入高版本的-->
<dependency>
   <groupId>javax.servlet</groupId>
   <artifactId>javax.servlet-api</artifactId>
   <version>4.0.1</version>
</dependency>
  • 注册bean:multipartResolver
<!--文件上传配置-->
    <bean id="multipartResolver"  class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!-- 请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-1 -->
        <property name="defaultEncoding" value="utf-8"/>
        <!-- 上传文件大小上限,单位为字节(10485760=10M) -->
        <property name="maxUploadSize" value="10485760"/>
        <property name="maxInMemorySize" value="40960"/>
    </bean>

测试

  • 编写请求响应
    @PostMapping("/insert")
    public String fileUpload(@RequestParam("file") CommonsMultipartFile file , HttpServletRequest request) throws IOException {
        //获取文件名 : file.getOriginalFilename();
        String uploadFileName = file.getOriginalFilename();
        //如果文件名为空,直接回到首页!
        if ("".equals(uploadFileName)){
            return "redirect:/index.jsp";
        }
        System.out.println("上传文件名 : "+uploadFileName);
        //上传路径保存设置
        String path = request.getServletContext().getRealPath("/upload");
        //如果路径不存在,创建一个
        File realPath = new File(path);
        if (!realPath.exists()){
            realPath.mkdir();
        }
        System.out.println("上传文件保存地址:"+realPath);
        InputStream is = file.getInputStream(); //文件输入流
        OutputStream os = new FileOutputStream(new File(realPath,uploadFileName)); //文件输出流
        //读取写出
        int len=0;
        byte[] buffer = new byte[1024];
        while ((len=is.read(buffer))!=-1){
            os.write(buffer,0,len);
            os.flush();
        }
        os.close();
        is.close();

        return "redirect:/index.jsp";
    }
    /*
     * 采用file.Transto 来保存上传的文件
     */
    @RequestMapping("/upload2")
    public String  fileUpload2(@RequestParam("file") CommonsMultipartFile file, HttpServletRequest request) throws IOException {

        //上传路径保存设置
        String path = request.getServletContext().getRealPath("/upload");
        File realPath = new File(path);
        if (!realPath.exists()){
            realPath.mkdir();
        }
        //上传文件地址
        System.out.println("上传文件保存地址:"+realPath);
        //通过CommonsMultipartFile的方法直接写文件(注意这个时候)
        file.transferTo(new File(realPath +"/"+ file.getOriginalFilename()));
        return "redirect:/index.jsp";
    }

文件下载

    @RequestMapping(value="/download")
    public String downloads(HttpServletResponse response ,HttpServletRequest request) throws Exception{
        //要下载的图片地址
        String  path = request.getServletContext().getRealPath("/upload");
        String  fileName = "01.jpg";
        //1、设置response 响应头
        response.reset(); //设置页面不缓存,清空buffer
        response.setCharacterEncoding("UTF-8"); //字符编码
        response.setContentType("multipart/form-data"); //二进制传输数据
        //设置响应头
        response.setHeader("Content-Disposition",
                "attachment;fileName="+ URLEncoder.encode(fileName, "UTF-8"));
        File file = new File(path,fileName);
        //2、 读取文件--输入流
        InputStream input=new FileInputStream(file);
        //3、 写出文件--输出流
        OutputStream out = response.getOutputStream();
        byte[] buff =new byte[1024];
        int index=0;
        //4、执行 写出操作
        while((index= input.read(buff))!= -1){
            out.write(buff, 0, index);
            out.flush();
        }
        out.close();
        input.close();
        return null;
    }

环境依赖配置

    <groupId>org.example</groupId>
    <artifactId>Spring</artifactId>
    <packaging>pom</packaging>
    <version>1.0-SNAPSHOT</version>
    <modules>
        <module>Spring-ioc</module>
        <module>Spring-MVC</module>
    </modules>


    <dependencies>
        <!--单元测试-->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.13</version>
            <scope>test</scope>
        </dependency>

        <!--mysql连接数据库-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.49</version>
        </dependency>

        <!--mybatis -->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis</artifactId>
            <version>3.5.5</version>
        </dependency>

        <!--spring-webmvc -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>5.2.8.RELEASE</version>
        </dependency>

        <!--spring-jdbc -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>5.2.8.RELEASE</version>
        </dependency>

        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.4</version>
        </dependency>

        <!--mybatis-spring -->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis-spring</artifactId>
            <version>2.0.5</version>
        </dependency>


        <!--lombok -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.12</version>
            <scope>provided</scope>
        </dependency>


        <!--servlet-api -->
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>4.0.1</version>
            <scope>provided</scope>
        </dependency>

        <!--JSTL表达式依赖-->
        <dependency>
            <groupId>javax.servlet.jsp.jstl</groupId>
            <artifactId>jstl-api</artifactId>
            <version>1.2</version>
        </dependency>
        <!--Standard标签库-->
        <dependency>
            <groupId>taglibs</groupId>
            <artifactId>standard</artifactId>
            <version>1.1.2</version>
        </dependency>
        <!--jsp-->
        <dependency>
            <groupId>javax.servlet.jsp</groupId>
            <artifactId>jsp-api</artifactId>
            <version>2.2</version>
        </dependency>
        <!--ackson-core-->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.11.1</version>
        </dependency>
                <!--jackson-databind -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.11.1</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-annotations -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-annotations</artifactId>
            <version>2.11.1</version>
        </dependency>
        <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-core -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.11.1</version>
        </dependency>
    </dependencies>

    <build>
        <resources>
            <resource>
                <directory>src/main/resources</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
            <resource>
                <directory>src/main/java</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
        </resources>
    </build>

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值