前言
SpringMVC的基本概念和工作流程。我们知道了MVC架构模式的重要性,以及如何使用SpringMVC来开发Web应用程序。希望这篇文章能够帮助你入门SpringMVC,并在实际项目中发挥作用。
一、什么是SpringMVC
SpringMVC是一种基于Java的Web框架,用于开发灵活、高效的Web应用程序。它是Spring框架的一部分,提供了一种模型-视图-控制器(MVC)的架构模式,用于将应用程序的不同部分分离开来,以实现更好的可维护性和可扩展性。
二、MVC架构模式
在理解SpringMVC之前,让我们先了解一下MVC架构模式。MVC代表模型(Model)、视图(View)和控制器(Controller)。这种模式将应用程序分为三个主要部分,每个部分都有不同的职责。
- 模型(Model):模型代表应用程序的数据和业务逻辑。它负责处理数据的读取、存储和操作。在SpringMVC中,模型通常是一个Java对象,用于封装数据。
- 视图(View):视图负责展示模型的数据给用户。它可以是一个HTML页面、一个XML文档或者其他任何形式的用户界面。在SpringMVC中,视图通常是一个JSP页面或者Thymeleaf模板。
- 控制器(Controller):控制器负责处理用户的请求,并根据请求的内容选择合适的模型和视图。它是应用程序的中心协调者,负责处理业务逻辑和控制流程。在SpringMVC中,控制器通常是一个Java类,使用注解或配置文件来定义请求映射和处理方法。
三、SpringMVC的工作流程
- 用户发来一个请求,首先进入的是前端控制器DispatcherServlet
- 前端控制器(DispacherServlet)将用户发来的请求发送给处理器映射器(HandlerMapping)
- 处理器映射器根据前端控制器发来的用户的请求找到对应符合的控制器(Handler),并且将其封装成处理器执行链,返回给前端控制器。
- 处理器适配器接收到来自前端控制器的执行链后,找到对应执行此执行链的处理器适配器(HandlerAdapter)来调用的具体的控制器(就是说其对应的方法或者逻辑)
- 处理器适配器 (HandlerAdaptoer) 会调用对应的具体的 Controller(处理业务逻辑)
- 控制器执行完成后,会返回一个ModelAndView对象给处理器适配器
- 处理器适配器将返回来的ModelAndView对象返回给前端控制器(到这里所有的业务处理过程就要完了,接下就是将结果以页面的的形式响应给用户)
- 前端控制器将返回回来的ModelAndView对象交给视图解析器(ViewResolver),视图解析器根据传过来的View对象解析成对应的页面对象
- ViewResolver 将封装好的将页面对象和Model对象返回给 DIspatcherServlet
- 前端控制器再将返回回来的对象交给视图(View)
- 视图根据传过来的Model对象再一次的对页面进行渲染(将模型数据填充至视图中),然后在返回给前端控制器。
- 前端控制器将完成的结果响应给浏览器,然后浏览器在展现给用户。
四、SpringMVC核心组件
- DispatcherServlet
DispatcherServlet是SpringMVC框架了里面的前端控制器
作用:统一处理用户发来的请求并和响应,相当于一个中间转换器,减少了各个组件之间的调度,减少的耦合性。
- HandlerMapping
HandlerMapping是SpringMVC框架里面的处理器映射器
作用:根据请求发来的url 和method找到对应的Handler(就是说在一个用到SpringMVC框架的项目中会有好多方法和逻辑,这个组件的作用就是找到对应的方法和组件返回给前端控制器)
- Handler
Handler处理器,注意,这个需由工程师自己开发。
作用:在 DispatcherServlet 的控制下,Handler对具体的用户请求进行处理。
- HandlerAdapter
HandlerAdapter是SpringMVC框架提供的处理器适配器
作用:根据映射器找到的处理器Handler信息,按照特定的规则去执行相关的处理器 Handler。
- ViewResolver
ViewResolver是SpringMVC框架提供的视图解析器
作用:就是字如其名,就是用来将处理的结果解析成视图来展现给用户。视图解析器根据逻辑视图名解析成物理视图名,生成View视图对象,最后对视图进行渲染响应给用户。
- View
View是开发者自己提供的视图
作用:根据model对象的要求来渲染页面,然后前端控器在响应给用户。
五、SpringMVC的优势
- 灵活性:SpringMVC提供了灵活的配置选项和扩展点,可以根据项目的需求进行定制。
- 可测试性:SpringMVC的代码结构清晰,易于编写单元测试和集成测试。
- 易于集成:SpringMVC可以与其他Spring框架(如Spring Boot)和第三方库(如Hibernate)无缝集成。
- 高效性:SpringMVC采用了基于注解的开发方式,减少了冗余的配置代码,提高了开发效率。
六、SpringMVC的配置与常用注解
在使用SpringMVC之前,需要进行一些配置工作。主要包括以下几个方面:
- 配置DispatcherServlet:在web.xml文件中配置DispatcherServlet,并指定它的URL映射。
- 配置HandlerMapping:通过注解或XML配置文件将处理器(Controller)映射到URL。
- 配置ViewResolver:通过注解或XML配置文件指定视图解析器的类型和位置。
SpringMVC提供了一些常用的注解,用于简化开发过程。以下是一些常用的注解及其作用:
- @Controller:标识一个类为处理器(Controller)。
- @RequestMapping:将请求URL映射到处理器的方法。
- @RequestParam:获取请求参数的值。
- @PathVariable:获取URL路径中的变量值。
- @ResponseBody:将方法的返回值直接作为响应体返回给客户端。
1.请求处理流程:
当客户端发送一个HTTP请求到服务器时,Spring MVC会按照以下步骤处理请求:
- 客户端发送请求到DispatcherServlet。
- DispatcherServlet根据请求的URL找到对应的处理器映射器(HandlerMapping)。
- 处理器映射器将请求映射到一个具体的处理器(Controller)。
- 处理器执行业务逻辑,并返回一个ModelAndView对象。
- DispatcherServlet将ModelAndView对象传递给视图解析器(ViewResolver)。
- 视图解析器根据ModelAndView中的视图名解析出一个具体的视图对象。
- DispatcherServlet将ModelAndView和视图对象传递给视图对象,进行视图渲染。
- 渲染后的视图生成响应,返回给客户端。
2.控制器的编写:
控制器是Spring MVC中负责处理请求的组件。编写控制器的步骤如下:
- 创建一个Java类,并使用@Controller注解标记该类为控制器。
- 在控制器类中定义处理请求的方法,使用@RequestMapping注解标记该方法对应的URL路径。
- 在方法中编写业务逻辑,并返回一个ModelAndView对象,该对象包含了处理结果和要渲染的视图名。
3.视图的渲染:
视图是用来展示处理结果的组件。Spring MVC支持多种视图技术,如JSP、Thymeleaf、Freemarker等。视图的渲染由视图解析器(ViewResolver)负责。视图解析器根据视图名解析出一个具体的视图对象,然后将ModelAndView对象和视图对象传递给视图对象进行渲染。
八、SpringMVC入门
1.添加pom.xml依赖
在视图解析时需要用到jstl+standard的这两个jar包
<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/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.example</groupId> <artifactId>mybatis_spring</artifactId> <packaging>war</packaging> <version>1.0-SNAPSHOT</version> <name>mybatis_spring Maven Webapp</name> <url>http://maven.apache.org</url> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <maven.compiler.source>1.8</maven.compiler.source> <maven.compiler.target>1.8</maven.compiler.target> <maven.compiler.plugin.version>3.7.0</maven.compiler.plugin.version> <!--添加jar包依赖--> <!--1.spring 5.0.2.RELEASE相关--> <spring.version>5.0.2.RELEASE</spring.version> <!-- jstl+standard --> <jstl.version>1.2</jstl.version> <standard.version>1.1.2</standard.version> <!-- spring --> <spring.version>5.0.2.RELEASE</spring.version> </properties> <dependencies> <!--1.spring相关--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-orm</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${spring.version}</version> </dependency> <!-- spring mvc相关依赖 --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>jstl</groupId> <artifactId>jstl</artifactId> <version>${jstl.version}</version> </dependency> <dependency> <groupId>taglibs</groupId> <artifactId>standard</artifactId> <version>${standard.version}</version> </dependency> </dependencies> <build> <finalName>mybatis_spring</finalName> </build> </project>
2.创建spring-mvc.xml
<?xml version="1.0" encoding="UTF-8"?> <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" xmlns="http://www.springframework.org/schema/beans" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd"> <!--1) 扫描com.zking.zf及子子孙孙包下的控制器(扫描范围过大,耗时) 配置了Spring框架的组件扫描功能,指定了需要扫描的包路径。Spring将会自动扫描这个包及其子包中的类,并将其注册为Spring的组件。 --> <context:component-scan base-package="com.ctb"/> <!--2) 此标签默认注册DefaultAnnotationHandlerMapping和AnnotationMethodHandlerAdapter --> <mvc:annotation-driven/> <!--3) 创建ViewResolver视图解析器 --> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <!-- viewClass需要在pom中引入两个包:standard.jar and jstl.jar --> <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"></property> <property name="prefix" value="/WEB-INF/jsp/"/> <property name="suffix" value=".jsp"/> </bean> <!--4) 单独处理图片、样式、js等资源 --> <mvc:resources location="/css/" mapping="/css/**"/> <mvc:resources location="/js/" mapping="/js/**"/> <mvc:resources location="WEB-INF/images/" mapping="/images/**"/> </beans>
详细描述:
配置了Spring框架的组件扫描功能,指定了需要扫描的包路径。Spring将会自动扫描这个包及其子包中的类,并将其注册为Spring的组件。
<context:component-scan base-package="com.ctb"/>
启用了Spring MVC的注解驱动功能,使得我们可以使用注解来配置和处理请求。
<mvc:annotation-driven/>
配置了一个视图解析器,用于解析视图的名称并返回相应的视图对象。这里使用的是InternalResourceViewResolver,它将会解析JSP视图。prefix属性指定了JSP文件的路径前缀,suffix属性指定了JSP文件的后缀。
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="viewClass" value="org.springframework.web.servlet.view.JstlView"></property> <property name="prefix" value="/WEB-INF/jsp/"> <property name="suffix" value=".jsp"> </property></property></bean>
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_3_1.xsd" version="3.1"> <display-name>Archetype Created Web Application</display-name> <!-- Spring和web项目集成start --> <!-- spring上下文配置文件 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring-context.xml</param-value> </context-param> <!-- 读取Spring上下文的监听器 --> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!-- Spring和web项目集成end --> <!-- 中文乱码处理 --> <filter> <filter-name>encodingFilter</filter-name> <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> <async-supported>true</async-supported> <init-param> <param-name>encoding</param-name> <param-value>UTF-8</param-value> </init-param> </filter> <filter-mapping> <filter-name>encodingFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <!-- Spring MVC servlet --> <servlet> <servlet-name>SpringMVC</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <!--此参数可以不配置,默认值为:/WEB-INF/springmvc-servlet.xml--> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:spring-mvc.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> <!--web.xml 3.0的新特性,是否支持异步--> <async-supported>true</async-supported> </servlet> <servlet-mapping> <servlet-name>SpringMVC</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
4.编写web层
package com.ctb.web; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; /** * @author 彪 * @site * @create 2023-09-04 */ @Controller @RequestMapping("/ctb") public class SpringmvcServlet { @RequestMapping("/hello") public String test(){ return "hello-shaniao"; } }
接下来我们就可以直接渲染到jsp页面即可
5.静态资源处理
所有静态资源文件都会被spring-mvc.xml配置文件进行拦截,所以要添加配置,让处理器进行扫描,专门处理图片、样式、js等资源...
<mvc:resources location="/css/" mapping="/css/**"/> <mvc:resources location="/js/" mapping="/js/**"/> <mvc:resources location="WEB-INF/images/" mapping="/images/**"/>