1. 初识Spring MVC

1.1 什么是Spring MVC

Spring MVC是Spring Framework的一部分,是基于 Java 实现 MVC 的轻量级 Web 框架。
Spring的 web 框架围绕 DispatcherServlet [ 调度Servlet ] 设计,DispatcherServlet 的作用是将请求分发到不同的处理器。从Spring 2.5开始,使用Java 5或者以上版本的用户可以采用基于注解形式进行开发,十分简洁;

1.2 前端控制器

SpringMVC的原理如下图所示:

Spring MVC - 图1

执行原理图
Spring MVC - 图2

  1. 用户发送请求至前端控制器 DispatcherServlet。
  2. DispatcherServlet 收到请求调用处理器映射器 HandlerMapping。
  3. 处理器映射器根据请求 url 找到具体的处理器,生成处理器执行链 HandlerExecutionChain(包括处理器对象和处理器拦截器)一并返回给 DispatcherServlet。
  4. DispatcherServlet 根据处理器 Handler 获取处理器适配器 HandlerAdapter 执行处理一系列的操作,如:参数封装,数据格式转换,数据验证等操作。
  5. 执行处理器 Handler(Controller,也叫页面控制器)。
  6. Handler 执行完成返回 ModelAndView。
  7. HandlerAdapter 将 Handler 执行结果 ModelAndView 返回到 DispatcherServlet
  8. DispatcherServlet 将 ModelAndView 传给 ViewReslover 视图解析器。
  9. ViewReslover 解析后返回具体 View
  10. DispatcherServlet 对 View 进行渲染视图(即将模型数据 model 填充至视图中)。
  11. DispatcherServlet 响应用户,将 View 最终视图呈现给用户。

1.3 快速上手

1、导入相关依赖;

  1. <dependencies>
  2. <!-- jackson -->
  3. <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-core -->
  4. <dependency>
  5. <groupId>com.fasterxml.jackson.core</groupId>
  6. <artifactId>jackson-databind</artifactId>
  7. <version>2.9.8</version>
  8. </dependency>
  9. <dependency>
  10. <groupId>org.springframework</groupId>
  11. <artifactId>spring-webmvc</artifactId>
  12. <version>5.2.5.RELEASE</version>
  13. </dependency>
  14. <dependency>
  15. <groupId>org.projectlombok</groupId>
  16. <artifactId>lombok</artifactId>
  17. <version>1.18.12</version>
  18. </dependency>
  19. <!-- servlet -->
  20. <dependency>
  21. <groupId>jstl</groupId>
  22. <artifactId>jstl</artifactId>
  23. <version>1.2</version>
  24. </dependency>
  25. <dependency>
  26. <groupId>javax.servlet</groupId>
  27. <artifactId>javax.servlet-api</artifactId>
  28. <version>4.0.0</version>
  29. </dependency>
  30. <!-- 实体类简写 -->
  31. <dependency>
  32. <groupId>org.projectlombok</groupId>
  33. <artifactId>lombok</artifactId>
  34. <version>1.18.12</version>
  35. </dependency>
  36. </dependencies>
  37. <build>
  38. <resources>
  39. <resource>
  40. <directory>src/main/java</directory>
  41. <includes>
  42. <include>**/*.properties</include>
  43. <include>**/*.xml</include>
  44. </includes>
  45. <filtering>false</filtering>
  46. </resource>
  47. <resource>
  48. <directory>src/main/resources</directory>
  49. <includes>
  50. <include>**/*.properties</include>
  51. <include>**/*.xml</include>
  52. </includes>
  53. <filtering>false</filtering>
  54. </resource>
  55. </resources>
  56. </build>

2、配置 web.xml 文件;

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  5. version="4.0">
  6. <!--1.注册servlet-->
  7. <servlet>
  8. <servlet-name>SpringMVC</servlet-name>
  9. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  10. <!--通过初始化参数指定SpringMVC配置文件的位置,进行关联-->
  11. <init-param>
  12. <param-name>contextConfigLocation</param-name>
  13. <param-value>classpath:spring-servlet.xml</param-value>
  14. </init-param>
  15. <!-- 启动顺序,数字越小,启动越早 -->
  16. <load-on-startup>1</load-on-startup>
  17. </servlet>
  18. <!--所有请求都会被springmvc拦截 -->
  19. <servlet-mapping>
  20. <servlet-name>SpringMVC</servlet-name>
  21. <url-pattern>/</url-pattern>
  22. </servlet-mapping>
  23. </web-app>

/ 和 /* 的区别:

  • /:不会匹配到.jsp, 只针对我们编写的请求;即:.jsp 不会进入spring的 DispatcherServlet类 。
  • /:会匹配 .jsp,会出现返回 jsp视图 时再次进入 spring 的 DispatcherServlet 类,导致找不到对应的 controller 所以报404错。

3、配置 spring-config.xml 配置文件

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:mvc="http://www.springframework.org/schema/mvc"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans.xsd
  8. http://www.springframework.org/schema/context
  9. https://www.springframework.org/schema/context/spring-context.xsd
  10. http://www.springframework.org/schema/mvc
  11. https://www.springframework.org/schema/mvc/spring-mvc.xsd">
  12. <!-- 自动扫描包,让指定包下的注解生效,由IOC容器统一管理 -->
  13. <context:component-scan base-package="com.xuwei.controller"/>
  14. <!-- 让Spring MVC不处理静态资源 -->
  15. <mvc:default-servlet-handler />
  16. <!--
  17. 支持mvc注解驱动
  18. 在spring中一般采用@RequestMapping注解来完成映射关系
  19. 要想使@RequestMapping注解生效
  20. 必须向上下文中注册DefaultAnnotationHandlerMapping
  21. 和一个AnnotationMethodHandlerAdapter实例
  22. 这两个实例分别在类级别和方法级别处理。
  23. 而annotation-driven配置帮助我们自动完成上述两个实例的注入。
  24. -->
  25. <mvc:annotation-driven />
  26. <!-- 视图解析器 -->
  27. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"
  28. id="internalResourceViewResolver">
  29. <!-- 前缀 -->
  30. <property name="prefix" value="/WEB-INF/jsp/" />
  31. <!-- 后缀 -->
  32. <property name="suffix" value=".jsp" />
  33. </bean>
  34. </beans>

注意:在视图解析器中我们把所有的视图都存放在/WEB-INF/目录下,这样可以保证视图安全,因为这个目录下的文件,客户端不能直接访问。

4、创建Controller,编写一个 Java 控制类:com.xuwei.controller.HelloController , 注意编码规范

  1. @Controller
  2. @RequestMapping("/HelloController")
  3. public class HelloController {
  4. //真实访问地址 : 项目名/HelloController/hello
  5. @RequestMapping("/hello")
  6. public String sayHello(Model model){
  7. //向模型中添加属性msg与值,可以在JSP页面中取出并渲染
  8. model.addAttribute("msg","hello,SpringMVC");
  9. //web-inf/jsp/hello.jsp
  10. return "hello";
  11. }
  12. }
  • @Controller 是为了让 Spring IOC 容器初始化时自动扫描到;
  • @RequestMapping 是为了映射请求路径,这里因为类与方法上都有映射所以访问时应该是 /HelloController/hello;
  • 方法中声明 Model 类型的参数是为了把 Action 中的数据带到视图中;
  • 方法返回的结果是视图的名称 hello,加上配置文件中的前后缀变成WEB-INF/jsp/hello.jsp。

5、创建视图层,在WEB-INF/ jsp 目录中创建 hello.jsp , 视图可以直接取出并展示从Controller带回的信息;也可以通过EL表示取出Model中存放的值,或者对象;

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>SpringMVC</title>
  5. </head>
  6. <body>
  7. ${msg}
  8. </body>
  9. </html>

2. RestFul风格

Restful 是一设置种风格。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。
传统方式操作资源 :通过不同的参数来实现不同的效果!方法单一,post 和 get

使用RESTful操作资源 :可以通过不同的请求方式来实现不同的效果!如下:请求地址一样,但是功能可以不同!


3. 结果跳转

3.1 ModelAndView、Model、ModelMap

SpringMVC 内部使用一个 org.springframework.ui.Model 接口存储的数据模型,功能类似于 Java.uitl.Map, 但是比 Map 更好用 org.springframework.ui.ModelMap 实现 Map 接口。

SpringMVC 在调用方法前会创建一个隐含的数据模型,作为模型数据的存储容器, 成为”隐含模型”。如果处理方法入参为 Map 或者 Model 类型,SpringMVC 会将隐含模型的引用传递给这些入参。

  • Model 是一个接口, 其实现类为 ExtendedModelMap,继承了 ModelMap 类。
  • ModelMap 对象主要用于传递控制方法处理数据到结果页面,也就是说我们把结果页面上需要的数据放到 ModelMap 对象中即可。ModelMap 或者 Model 通过 addAttribute 方法向页面传递参数。注意:modelmap 本身不能设置页面跳转的url地址别名或者物理跳转地址
  • ModelAndView 对象有两个作用:

    • 设置转向地址,这也是 ModelAndView 和 ModelMap 的主要区别,通过 setViewName 方法;
    • 通过 addObject 方法可以将结果页面需要的数据放到 ModelAndView 对象中,类似于request对象的setAttribute方法的作用,用来在一个请求过程中传递处理的数据。

      3.2 转发和重定向

      1. @Controller
      2. public class ResultSpringMVC2 {
      3. @RequestMapping("/rsm2/t1")
      4. public String test1(){
      5. //转发
      6. return "test";
      7. }
      8. @RequestMapping("/rsm2/t2")
      9. public String test2(){
      10. //重定向
      11. return "redirect:/index.jsp";
      12. //return "redirect:hello.do"; //hello.do为另一个请求/
      13. }

4. 解决乱码问题

4.1 Get 请求乱码解决

1、修改 tomcat 配置文件 :设置编码!

  1. <Connector URIEncoding="utf-8" port="8080" protocol="HTTP/1.1"
  2. connectionTimeout="20000"
  3. redirectPort="8443" />

2、自定义过滤器

  1. package com.kuang.filter;
  2. import javax.servlet.*;
  3. import javax.servlet.http.HttpServletRequest;
  4. import javax.servlet.http.HttpServletRequestWrapper;
  5. import javax.servlet.http.HttpServletResponse;
  6. import java.io.IOException;
  7. import java.io.UnsupportedEncodingException;
  8. import java.util.Map;
  9. /**
  10. * 解决get和post请求 全部乱码的过滤器
  11. */
  12. public class GenericEncodingFilter implements Filter {
  13. @Override
  14. public void destroy() {
  15. }
  16. @Override
  17. public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
  18. //处理response的字符编码
  19. HttpServletResponse myResponse=(HttpServletResponse) response;
  20. myResponse.setContentType("text/html;charset=UTF-8");
  21. // 转型为与协议相关对象
  22. HttpServletRequest httpServletRequest = (HttpServletRequest) request;
  23. // 对request包装增强
  24. HttpServletRequest myrequest = new MyRequest(httpServletRequest);
  25. chain.doFilter(myrequest, response);
  26. }
  27. @Override
  28. public void init(FilterConfig filterConfig) throws ServletException {
  29. }
  30. }
  31. //自定义request对象,HttpServletRequest的包装类
  32. class MyRequest extends HttpServletRequestWrapper {
  33. private HttpServletRequest request;
  34. //是否编码的标记
  35. private boolean hasEncode;
  36. //定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
  37. public MyRequest(HttpServletRequest request) {
  38. super(request);// super必须写
  39. this.request = request;
  40. }
  41. // 对需要增强方法 进行覆盖
  42. @Override
  43. public Map getParameterMap() {
  44. // 先获得请求方式
  45. String method = request.getMethod();
  46. if (method.equalsIgnoreCase("post")) {
  47. // post请求
  48. try {
  49. // 处理post乱码
  50. request.setCharacterEncoding("utf-8");
  51. return request.getParameterMap();
  52. } catch (UnsupportedEncodingException e) {
  53. e.printStackTrace();
  54. }
  55. } else if (method.equalsIgnoreCase("get")) {
  56. // get请求
  57. Map<String, String[]> parameterMap = request.getParameterMap();
  58. if (!hasEncode) { // 确保get手动编码逻辑只运行一次
  59. for (String parameterName : parameterMap.keySet()) {
  60. String[] values = parameterMap.get(parameterName);
  61. if (values != null) {
  62. for (int i = 0; i < values.length; i++) {
  63. try {
  64. // 处理get乱码
  65. values[i] = new String(values[i]
  66. .getBytes("ISO-8859-1"), "utf-8");
  67. } catch (UnsupportedEncodingException e) {
  68. e.printStackTrace();
  69. }
  70. }
  71. }
  72. }
  73. hasEncode = true;
  74. }
  75. return parameterMap;
  76. }
  77. return super.getParameterMap();
  78. }
  79. //取一个值
  80. @Override
  81. public String getParameter(String name) {
  82. Map<String, String[]> parameterMap = getParameterMap();
  83. String[] values = parameterMap.get(name);
  84. if (values == null) {
  85. return null;
  86. }
  87. return values[0]; // 取回参数的第一个值
  88. }
  89. //取所有值
  90. @Override
  91. public String[] getParameterValues(String name) {
  92. Map<String, String[]> parameterMap = getParameterMap();
  93. String[] values = parameterMap.get(name);
  94. return values;
  95. }
  96. }

4.2 Post 请求乱码解决

可以在 web.xml 中配置 Spring 自带的过滤器。

  1. <filter>
  2. <filter-name>encoding</filter-name>
  3. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  4. <init-param>
  5. <param-name>encoding</param-name>
  6. <param-value>utf-8</param-value>
  7. </init-param>
  8. </filter>
  9. <filter-mapping>
  10. <filter-name>encoding</filter-name>
  11. <url-pattern>/*</url-pattern>
  12. </filter-mapping>

5. JSON交互

5.1 JSON 使用

要实现从JSON字符串转换为JavaScript 对象,使用 JSON.parse() 方法:

  1. var obj = JSON.parse('{"a": "Hello", "b": "World"}');
  2. //结果是 {a: 'Hello', b: 'World'}

要实现从JavaScript 对象转换为JSON字符串,使用 JSON.stringify() 方法:

  1. var json = JSON.stringify({a: 'Hello', b: 'World'});
  2. //结果是 '{"a": "Hello", "b": "World"}'

5.2 使用 Jackson

1、导入 jackson 依赖。

  1. <!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-core -->
  2. <dependency>
  3. <groupId>com.fasterxml.jackson.core</groupId>
  4. <artifactId>jackson-databind</artifactId>
  5. <version>2.9.8</version>
  6. </dependency>

2、配置 web.xml。

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
  5. version="4.0">
  6. <!--1.注册servlet-->
  7. <servlet>
  8. <servlet-name>SpringMVC</servlet-name>
  9. <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  10. <!--通过初始化参数指定SpringMVC配置文件的位置,进行关联-->
  11. <init-param>
  12. <param-name>contextConfigLocation</param-name>
  13. <param-value>classpath:springmvc-servlet.xml</param-value>
  14. </init-param>
  15. <!-- 启动顺序,数字越小,启动越早 -->
  16. <load-on-startup>1</load-on-startup>
  17. </servlet>
  18. <!--所有请求都会被springmvc拦截 -->
  19. <servlet-mapping>
  20. <servlet-name>SpringMVC</servlet-name>
  21. <url-pattern>/</url-pattern>
  22. </servlet-mapping>
  23. <filter>
  24. <filter-name>encoding</filter-name>
  25. <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
  26. <init-param>
  27. <param-name>encoding</param-name>
  28. <param-value>utf-8</param-value>
  29. </init-param>
  30. </filter>
  31. <filter-mapping>
  32. <filter-name>encoding</filter-name>
  33. <url-pattern>/</url-pattern>
  34. </filter-mapping>
  35. </web-app>

3、配置 spring-config.xml。

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:mvc="http://www.springframework.org/schema/mvc"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans.xsd
  8. http://www.springframework.org/schema/context
  9. https://www.springframework.org/schema/context/spring-context.xsd
  10. http://www.springframework.org/schema/mvc
  11. https://www.springframework.org/schema/mvc/spring-mvc.xsd">
  12. <!-- 自动扫描指定的包,下面所有注解类交给IOC容器管理 -->
  13. <context:component-scan base-package="com.kuang.controller"/>
  14. <!-- 视图解析器 -->
  15. <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"
  16. id="internalResourceViewResolver">
  17. <!-- 前缀 -->
  18. <property name="prefix" value="/WEB-INF/jsp/" />
  19. <!-- 后缀 -->
  20. <property name="suffix" value=".jsp" />
  21. </bean>
  22. </beans>

4、编写 User 实体类。

  1. @Data
  2. @AllArgsConstructor
  3. @NoArgsConstructor
  4. public class User {
  5. private String name;
  6. private int age;
  7. private String sex;
  8. }

5、编写 Controller。

  1. @Controller
  2. public class UserController {
  3. @RequestMapping("/json1")
  4. @ResponseBody
  5. public String json1() throws JsonProcessingException {
  6. //创建一个jackson的对象映射器,用来解析数据
  7. ObjectMapper mapper = new ObjectMapper();
  8. //创建一个对象
  9. User user = new User("秦疆1号", 3, "男");
  10. //将我们的对象解析成为json格式
  11. String str = mapper.writeValueAsString(user);
  12. //由于@ResponseBody注解,这里会将str转成json格式返回;十分方便
  13. return str;
  14. }
  15. }

6、启动测试,发现乱码!
Spring MVC - 图3

7、通过 @RequestMapping 的 produces 属性来实现,修改下代码

  1. //produces:指定响应体返回类型和编码
  2. @RequestMapping(value = "/json1",produces = "application/json;charset=utf-8")

8、如果每次定义方法都要定义 produces 属性的话,太麻烦了,乱码统一解决,在 spring-config 中配置:

  1. <mvc:annotation-driven>
  2. <mvc:message-converters register-defaults="true">
  3. <bean class="org.springframework.http.converter.StringHttpMessageConverter">
  4. <constructor-arg value="UTF-8"/>
  5. </bean>
  6. <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
  7. <property name="objectMapper">
  8. <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
  9. <property name="failOnEmptyBeans" value="false"/>
  10. </bean>
  11. </property>
  12. </bean>
  13. </mvc:message-converters>
  14. </mvc:annotation-driven>

9、我们可以在类上使用注解 @RestController,表示该类所有的方法都只会返回 json 字符串了。


6. Ajax 处理

6.1 认识 Ajax

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

6.2 使用 Ajax

1、配置 web.xml 和 spring-config.xml,这里省略。
2、编写一个 AjaxController。

  1. @RequestMapping("/a3")
  2. public String ajax3(String name,String pwd){
  3. String msg = "";
  4. //模拟数据库中存在数据
  5. if (name!=null){
  6. if ("admin".equals(name)){
  7. msg = "OK";
  8. }else {
  9. msg = "用户名输入错误";
  10. }
  11. }
  12. if (pwd!=null){
  13. if ("123456".equals(pwd)){
  14. msg = "OK";
  15. }else {
  16. msg = "密码输入有误";
  17. }
  18. }
  19. return msg; //由于@RestController注解,将msg转成json格式返回
  20. }

3、前端页面 login.jsp

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>ajax</title>
  5. <script src="${pageContext.request.contextPath}/statics/js/jquery-3.1.1.min.js"></script>
  6. <script>
  7. function a1(){
  8. $.post({
  9. url:"${pageContext.request.contextPath}/a3",
  10. data:{'name':$("#name").val()},
  11. success:function (data) {
  12. if (data.toString()=='OK'){
  13. $("#userInfo").css("color","green");
  14. }else {
  15. $("#userInfo").css("color","red");
  16. }
  17. $("#userInfo").html(data);
  18. }
  19. });
  20. }
  21. function a2(){
  22. $.post({
  23. url:"${pageContext.request.contextPath}/a3",
  24. data:{'pwd':$("#pwd").val()},
  25. success:function (data) {
  26. if (data.toString()=='OK'){
  27. $("#pwdInfo").css("color","green");
  28. }else {
  29. $("#pwdInfo").css("color","red");
  30. }
  31. $("#pwdInfo").html(data);
  32. }
  33. });
  34. }
  35. </script>
  36. </head>
  37. <body>
  38. <p>
  39. 用户名:<input type="text" id="name" onblur="a1()"/>
  40. <span id="userInfo"></span>
  41. </p>
  42. <p>
  43. 密码:<input type="text" id="pwd" onblur="a2()"/>
  44. <span id="pwdInfo"></span>
  45. </p>
  46. </body>
  47. </html>

7. 拦截器

7.1 什么是拦截器

SpringMVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter, 用于对处理器进行预处理和后处理。开发者可以自己定义一些拦截器来实现特定的功能。

过滤器与拦截器的区别:拦截器是 AOP 思想的具体应用。
过滤器

  • servlet规范中的一部分,任何 java web 工程都可以使用
  • 在url-pattern中配置了/*之后,可以对所有要访问的资源进行拦截

拦截器

  • 拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能使用
  • 拦截器只会拦截访问的控制器方法, 如果访问的是jsp/html/css/image/js是不会进行拦截的

    7.2 自定义拦截器(验证用户是否登录)

    1、编写一个登录页面 login.jsp。

    1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    2. <html>
    3. <head>
    4. <title>Title</title>
    5. </head>
    6. <h1>登录页面</h1>
    7. <hr>
    8. <body>
    9. <form action="${pageContext.request.contextPath}/user/login">
    10. 用户名:<input type="text" name="username"> <br>
    11. 密码:<input type="password" name="pwd"> <br>
    12. <input type="submit" value="提交">
    13. </form>
    14. </body>
    15. </html>

2、编写一个 Controller 处理请求。

  1. @Controller
  2. @RequestMapping("/user")
  3. public class UserController {
  4. //跳转到登陆页面
  5. @RequestMapping("/jumplogin")
  6. public String jumpLogin() throws Exception {
  7. return "login";
  8. }
  9. //跳转到成功页面
  10. @RequestMapping("/jumpSuccess")
  11. public String jumpSuccess() throws Exception {
  12. return "success";
  13. }
  14. //登陆提交
  15. @RequestMapping("/login")
  16. public String login(HttpSession session, String username, String pwd) throws Exception {
  17. // 向session记录用户身份信息
  18. System.out.println("接收前端==="+username);
  19. session.setAttribute("user", username);
  20. return "success";
  21. }
  22. //退出登陆
  23. @RequestMapping("logout")
  24. public String logout(HttpSession session) throws Exception {
  25. // session 过期
  26. session.invalidate();
  27. return "login";
  28. }
  29. }

3、编写一个登录成功的页面 success.jsp。

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>Title</title>
  5. </head>
  6. <body>
  7. <h1>登录成功页面</h1>
  8. <hr>
  9. ${user}
  10. <a href="${pageContext.request.contextPath}/user/logout">注销</a>
  11. </body>
  12. </html>

4、在 index 页面上测试跳转!启动 tomcat 测试,没有登录也可以进入主页!

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>$Title$</title>
  5. </head>
  6. <body>
  7. <h1>首页</h1>
  8. <hr>
  9. <%--登录--%>
  10. <a href="${pageContext.request.contextPath}/user/jumplogin">登录</a>
  11. <a href="${pageContext.request.contextPath}/user/jumpSuccess">成功页面</a>
  12. </body>
  13. </html>

5、编写用户登录拦截器。

  1. public class LoginInterceptor implements HandlerInterceptor {
  2. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws ServletException, IOException {
  3. // 如果是登陆页面则放行
  4. System.out.println("uri: " + request.getRequestURI());
  5. if (request.getRequestURI().contains("login")) {
  6. return true;
  7. }
  8. HttpSession session = request.getSession();
  9. // 如果用户已登陆也放行
  10. if(session.getAttribute("user") != null) {
  11. return true;
  12. }
  13. // 用户没有登陆跳转到登陆页面
  14. request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request, response);
  15. return false;
  16. }
  17. public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {
  18. }
  19. public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {
  20. }
  21. }

6、在 spring 的配置文件中注册拦截器

  1. <!--关于拦截器的配置-->
  2. <mvc:interceptors>
  3. <mvc:interceptor>
  4. <mvc:mapping path="/**"/>
  5. <bean id="loginInterceptor" class="com.kuang.interceptor.LoginInterceptor"/>
  6. </mvc:interceptor>
  7. </mvc:interceptors>

8. 文件上传

8.1 使用 IO 流写文件

1、导入文件上传的 jar 包。

  1. <!--文件上传-->
  2. <dependency>
  3. <groupId>commons-fileupload</groupId>
  4. <artifactId>commons-fileupload</artifactId>
  5. <version>1.3.3</version>
  6. </dependency>
  7. <!--servlet-api导入高版本的-->
  8. <dependency>
  9. <groupId>javax.servlet</groupId>
  10. <artifactId>javax.servlet-api</artifactId>
  11. <version>4.0.1</version>
  12. </dependency>

2、配置 bean:multipartResolver

这个bena的id必须为:multipartResolver , 否则上传文件会报400的错误

  1. <!-- 文件上传-->
  2. <bean id="multipartResolver"
  3. class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
  4. <!-- 请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-1 -->
  5. <property name="defaultEncoding" value="utf-8"/>
  6. <!-- 上传文件大小上限,单位为字节(10485760=10M) -->
  7. <property name="maxUploadSize" value="10485760"/>
  8. <property name="maxInMemorySize" value="40960"/>
  9. </bean>

3、编写前端页面。

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <html>
  3. <head>
  4. <title>$Title$</title>
  5. </head>
  6. <body>
  7. <h3>hello world</h3>
  8. <form action="/upload" enctype="multipart/form-data" method="post">
  9. <input type="file" name="file"/>
  10. <input type="submit" value="upload">
  11. </form>
  12. </body>
  13. </html>

4、编写 Controller

  1. @Controller
  2. public class FileController {
  3. //@RequestParam("file") 将name=file控件得到的文件封装成CommonsMultipartFile 对象
  4. @RequestMapping("/upload")
  5. public String fileUpload(@RequestParam("file")CommonsMultipartFile file,
  6. HttpServletRequest request) throws IOException {
  7. //获取文件名
  8. String uploadFileName = file.getOriginalFilename();
  9. //如果文件名为空,直接回到首页
  10. if ("".equals(uploadFileName)) {
  11. return "redirect:/index.jsp";
  12. }
  13. //上传路径保存设置
  14. String path = request.getServletContext().getRealPath("/upload");
  15. //如果路径不存在,创建一个
  16. File realPath = new File(path);
  17. if (! realPath.exists()) {
  18. realPath.mkdir();
  19. }
  20. System.out.println("上传文件保存地址:" + realPath);
  21. InputStream is = file.getInputStream(); //文件输入流
  22. OutputStream os = new FileOutputStream(new File(realPath,uploadFileName)); //文件输出流
  23. //读取写出
  24. int len=0;
  25. byte[] buffer = new byte[1024];
  26. while ((len=is.read(buffer))!=-1){
  27. os.write(buffer,0,len);
  28. os.flush();
  29. }
  30. os.close();
  31. is.close();
  32. return "redirect:/index.jsp";
  33. }
  34. }

5、测试上传文件,OK!

8.2 采用 file.Transto 来保存上传的文件

1、编写Controller

  1. /*
  2. * 采用file.Transto 来保存上传的文件
  3. */
  4. @RequestMapping("/upload2")
  5. public String fileUpload2(@RequestParam("file") CommonsMultipartFile file, HttpServletRequest request) throws IOException {
  6. //上传路径保存设置
  7. String path = request.getServletContext().getRealPath("/upload");
  8. File realPath = new File(path);
  9. if (!realPath.exists()){
  10. realPath.mkdir();
  11. }
  12. //上传文件地址
  13. System.out.println("上传文件保存地址:"+realPath);
  14. //通过CommonsMultipartFile的方法直接写文件(注意这个时候)
  15. file.transferTo(new File(realPath +"/"+ file.getOriginalFilename()));
  16. return "redirect:/index.jsp";
  17. }

2、前端表单提交地址修改
3、访问提交测试,OK!

8.3 文件下载

1、编写 controller

  1. @RequestMapping(value="/download")
  2. public String downloads(HttpServletResponse response ,HttpServletRequest request) throws Exception{
  3. //要下载的图片地址
  4. String path = request.getServletContext().getRealPath("/upload");
  5. String fileName = "基础语法.jpg";
  6. //1、设置response 响应头
  7. response.reset(); //设置页面不缓存,清空buffer
  8. response.setCharacterEncoding("UTF-8"); //字符编码
  9. response.setContentType("multipart/form-data"); //二进制传输数据
  10. //设置响应头
  11. response.setHeader("Content-Disposition",
  12. "attachment;fileName="+URLEncoder.encode(fileName, "UTF-8"));
  13. File file = new File(path,fileName);
  14. //2、 读取文件--输入流
  15. InputStream input=new FileInputStream(file);
  16. //3、 写出文件--输出流
  17. OutputStream out = response.getOutputStream();
  18. byte[] buff =new byte[1024];
  19. int index=0;
  20. //4、执行 写出操作
  21. while((index= input.read(buff))!= -1){
  22. out.write(buff, 0, index);
  23. out.flush();
  24. }
  25. //5、关闭流
  26. out.close();
  27. input.close();
  28. return null;
  29. }

2、前端 index.jsp

  1. <a href="/download">点击下载</a>