SpringMVC执行原理
1. 第一个SpringMVC程序
1.1 新建一个父项目
新建一个普通maven项目,并导入依赖
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com</groupId>
<artifactId>SpringMVC-Father</artifactId>
<packaging>pom</packaging>
<version>1.0-SNAPSHOT</version>
<modules>
<module>SringMVC01-Servlet</module>
<module>SpringMVC02-son-HelloMVC</module>
</modules>
<dependencies>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.10</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
<!--Spring以及SpringWeb相关包 https://mvnrepository.com/artifact/org.springframework/spring-web -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/javax.servlet.jsp/jsp-api -->
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/javax.servlet/jstl -->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/javax.servlet/servlet-api -->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
</dependencies>
</project>
1.2 新建一个子项目,并添加webapp框架支持。
如果是在IDEA中开发,先新建一个子项目,然后右击子项目,添加框架框架支持,加入web支持。完整的项目截图为
1.3 web.xml
<?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">
<!-- 1. 注册DispatcherServlet-->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 关联一个SpringMVC的配置文件:【】 -->
<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>
<!-- / 匹配所有的请求:(不包括.jsp)
/* 匹配所有的请求:包括.jsp-->
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
1.4 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"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
<!--添加处理映射器-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
<!-- 添加处理器适配器-->
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
<!-- 添加视图解析器:DispatcherServlet给它的ModelAndView-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<!-- 前缀后缀-->
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
<!-- 将我们写的控制器配置在IOC里面-->
<bean id="/hello" class="com.controller.HelloController"/>
</beans>
1.5 HelloController.java
package com.controller;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class HelloController implements Controller {
public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
//ModelAndView 模型和视图
ModelAndView modelAndView = new ModelAndView();
//这里执行我们的具体的业务代码即可,也是开发中需要关注的重点
//封装对象,放在这个模型和视图中。
modelAndView.addObject("msg", "helloSpringMVC!");
//封装要跳转的视图,放在这个对象中
modelAndView.setViewName("hello");//在视图解析器中,我们配置了前缀和后缀
return modelAndView;
}
}
1.6 hello.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
${msg}
</body>
</html>
1.7 运行截图
注意:如果请求结果为404,有可能的原因是因为IDEA环境的原因,我们需要在项目结构的发布路径下面,手动建立一个lib包,将项目依赖添加到发布路径里面
2. 运行原理
2.1 官方版的原理图。
2.2 详解图。
2.3 DispatcherServlet中央控制器
它也是前端控制器,是整个SpringMVC的控制中心,当用户发出请求,DispatcherServlet接受请求并切拦截请求,在web.xml中,我们只配置了一个DispatcherServlet(它的本质还是Servlet,包括SpringMVC的底层都是Servlet),注意这个东西必须要绑定一个SpringMVC的配置文件。
<!-- 1. 注册DispatcherServlet-->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 关联一个SpringMVC的配置文件:【】 -->
<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>
<!-- / 匹配所有的请求:(不包括.jsp)
/* 匹配所有的请求:包括.jsp-->
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
2.4 HandlerMapping处理映射器
它被DispatcherServlet调用,并根据请求的url查找Handler(Controller),即在上面的url中http://localhost:8080/hello解析出hello
<!--添加处理映射器-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>
2.5 HandlerExecution
将解析后的信息传递给DispatcherServlet,如解析控制器映射等等。
2.6 HandlerAdapter处理器适配器
按照特定的规则去执行Handler
<!-- 添加处理器适配器-->
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
2.7 Handler处理器
Handler让具体的Controller去执行,Controller中的执行代码,是需要我们自己写的,Controller中需要返回一个ModelAndView对象(实行了Controller接口)
<!-- Handler-->
<bean id="/hello" class="com.controller.HelloController"/>
2.8 ModelAndView模型和视图
顾名思义,这个对象包含了两个至少需要填充的字段,模型,视图。模型是我们需要返回给前端的一些数据,视图是这个数据要渲染到哪个视图。它经过Controller中返回之后,由经handler和HandlerAdapter返回到DispatcherServlet。
public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
//ModelAndView 模型和视图
ModelAndView modelAndView = new ModelAndView();
//封装对象,放在这个模型和视图中。
modelAndView.addObject("msg", "helloSpringMVC!");
//封装要跳转的视图,放在这个对象中
modelAndView.setViewName("hello");//在配置文件中,我们配置了前缀和后缀
return modelAndView;
}
2.9 ViewResolver视图解析器
在ModelAndView被返回到DispatcherServlet中之后,DispatcherServlet会去调用ViewResolver来解析HandlerAdapter返回的逻辑视图名。
<!-- 添加视图解析器:DispatcherServlet给它的ModelAndView
1.获取了ModelAndView中的数据
2.解析ModelAndView中的视图名字
3.拼接视图名字,找到视图(但是这里基本不需要我们操作。)
4.将数据渲染到这个视图-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver">
<!-- 前缀后缀-->
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
在实际开发中,我们一般只关注Controller中的业务代码。