目录
一、概述
( 1 ) 是什么
( 2 ) 作用
二、工作流程
( 1 ) 流程
( 2 ) 步骤
三、入门实例
( 1 ) 入门实例
( 2 ) 静态资源处理
给我们带来的收获
一、概述
( 1 ) 是什么
SpringMVC是一个基于Java的Web应用开发框架,它是Spring Framework的一部分。它提供了一种模型(Model)-视图(View)-控制器(Controller)(MVC)的架构模式,用于构建灵活、可扩展的Web应用程序。SpringMVC通过将应用程序分解为模型(数据)、视图(用户界面)和控制器(处理请求和响应)三个组件,实现了松耦合和高度可测试的开发方式。它还提供了许多功能,如请求映射、数据绑定、表单处理、验证、国际化等,以简化Web开发过程。SpringMVC还支持各种视图技术,如JSP、Thymeleaf、Freemarker等,使开发人员能够选择适合自己的视图模板。总的来说,SpringMVC是一个强大而灵活的框架,把复杂的web应用分成逻辑清晰的几部分,简化开发,减少出错,方便组内开发人员之间的配合,广泛用于构建Java Web应用程序。
( 2 ) 作用
SpringMVC具有以下几个主要作用:
- 1. 分离关注点:SpringMVC通过将应用程序分解为模型、视图和控制器三个组件,实现了关注点的分离。模型负责处理数据,视图负责展示用户界面,控制器负责处理请求和响应。这种分离使得开发人员可以专注于各自的领域,提高了代码的可读性和可维护性。
- 2. 灵活可扩展:SpringMVC采用了松耦合的设计,使得开发人员可以根据需求选择合适的组件和技术进行开发。它支持各种视图技术,如JSP、Thymeleaf、Freemarker等,使开发人员能够选择适合自己的视图模板。同时,SpringMVC还提供了丰富的功能和特性,如拦截器、数据绑定、表单处理、验证、异常处理等,以满足不同场景下的需求。
- 3. 易于测试:由于SpringMVC采用了MVC架构模式,使得应用程序的各个组件可以独立测试。开发人员可以针对模型、视图和控制器编写单元测试,验证其功能和逻辑的正确性。这种可测试性有助于提高代码质量和开发效率。
- 4. 提高开发效率:SpringMVC提供了一套简洁而强大的API和工具,使开发人员能够快速构建Web应用程序。它通过注解和约定大于配置的方式,简化了开发过程,减少了样板代码的编写。同时,SpringMVC还提供了丰富的文档和社区支持,使开发人员能够快速解决问题和获取帮助。
总的来说,SpringMVC作为一个Web应用开发框架,具有分离关注点、灵活可扩展、易于测试和提高开发效率等作用。它帮助开发人员构建高效、易于维护的Web应用程序,并提供了丰富的功能和特性,以满足不同场景下的需求。
二、工作流程
( 1 ) 流程
当客户端发送请求到SpringMVC应用程序时,请求将由前端控制器(Front Controller)DispatcherServlet接收。DispatcherServlet是SpringMVC的核心组件,负责协调和管理请求的处理流程。
DispatcherServlet根据请求的URL找到对应的处理器(Handler),Handler是一个带有注解的方法或类,用于处理请求并生成响应。处理器可以根据请求的特征,如URL路径、请求方法、请求参数等进行匹配。
在执行处理器之前,SpringMVC通过处理器适配器(HandlerAdapter)对处理器进行封装和适配,使其符合统一的处理器接口(HandlerInterface)。这样就可以支持不同类型的处理器,如注解控制器(Annotation Controller)、Controller接口实现类、REST控制器等。
处理器执行过程中可以通过模型(Model)对象传递数据,模型是一个数据容器,用于保存处理结果并传递给视图(View)。处理器还可以返回一个逻辑视图名,告诉DispatcherServlet选择哪个视图来渲染响应。
DispatcherServlet根据逻辑视图名选择合适的视图解析器(ViewResolver)来解析视图,并生成最终的响应。视图解析器负责将逻辑视图名解析为具体的视图,如JSP、Thymeleaf模板等。视图负责生成最终的HTML、XML、JSON等响应内容。
最后,DispatcherServlet将响应发送给客户端进行展示。整个请求-响应过程通过中央调度器DispatcherServlet进行协调和控制,使开发者能够集中精力在业务逻辑的处理上,提高开发效率。
SpringMVC还提供了许多功能和特性,如拦截器(Interceptor)用于对请求进行预处理和后处理、数据绑定和类型转换、表单验证、异常处理、RESTful支持等,以满足不同场景下的需求。
总结而言,SpringMVC是一个基于Java的Web应用开发框架,通过MVC架构模式实现了灵活、可扩展的Web应用程序开发。它提供了一套全面的特性和组件,使开发人员能够快速构建高效、易于维护的Web应用程序。
( 2 ) 步骤
当客户端发送请求到SpringMVC应用程序时,请求会经过以下详细的流程:
- 1. 客户端发送请求:客户端通过浏览器发送HTTP请求,其中包含请求的URL、请求方法、请求头、请求参数等信息。
- 2. 前端控制器接收请求:请求到达应用程序后,被前端控制器DispatcherServlet接收。DispatcherServlet是一个Servlet,是整个SpringMVC应用的入口点。
- 3. 处理器映射器进行处理器匹配:DispatcherServlet通过处理器映射器(HandlerMapping)来确定请求对应的处理器(Handler)。处理器映射器根据配置的URL映射规则,将请求URL映射到相应的处理器。
- 4. 处理器适配器处理处理器:处理器适配器(HandlerAdapter)根据处理器的类型调用相应的处理方法。处理器可以是注解控制器(如@Controller、@RequestMapping)、实现了Controller接口的类、RESTful控制器等。
- 5. 处理器执行:处理器执行具体的业务逻辑,处理请求并生成一个Model(模型)对象。Model对象可以携带处理结果和数据,供视图渲染使用。
- 6. 处理器拦截器的调用:在处理器执行前后,可以定义拦截器对请求进行拦截和处理。拦截器可以在请求前进行预处理,也可以在请求后进行后处理。拦截器的执行顺序可以通过配置进行控制。
- 7. 处理器返回视图名或视图对象:处理器执行完毕后,会返回一个逻辑视图名或直接返回一个具体的视图对象。逻辑视图名是一个表示视图的逻辑名称,视图对象则代表具体的视图实现。
- 8. 视图解析器进行视图解析:DispatcherServlet通过视图解析器(ViewResolver)将逻辑视图名解析为具体的视图实现。视图解析器根据配置的规则来查找对应的视图,如JSP、Thymeleaf等。
- 9. 视图渲染:视图负责生成最终的响应内容,根据模型数据渲染视图模板,并生成最终的HTML、XML、JSON等响应。
- 10. 视图返回响应:渲染完成后,视图将生成的响应返回给前端控制器DispatcherServlet。
- 11. 处理器拦截器的后处理:在视图返回响应后,可以继续执行拦截器的后处理逻辑。拦截器可以对响应进行进一步的处理,如日志记录、跳转等。
- 12. 响应发送给客户端:DispatcherServlet将响应发送给客户端,客户端根据响应类型进行展示,如在浏览器中渲染HTML页面或解析JSON数据。
整个SpringMVC的工作流程是一个基于Servlet的请求-响应模型,通过前端控制器、处理器映射器、处理器适配器、视图解析器和视图等组件的协作,实现了灵活、可扩展并且易于开发的Web应用框架。DispatcherServlet充当了中央调度器的角色,协调和控制请求的处理流程。通过这种流程,SpringMVC实现了松耦合、可扩展和易于测试的Web应用开发。
三、入门实例
这里是根据我博客中的文章进行了一个扩展增加代码 : 认识Mybatis的关联关系映射
( 1 ) 入门实例
在项目中的pom.xml文件中添加依赖,增加后的所有引用配置如下 :
<?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>org.example</groupId>
<artifactId>index</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<name>index Maven Webapp</name>
<!-- FIXME change it to the project's website -->
<url>http://www.example.com</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>
<!--2.mybatis相关-->
<mybatis.version>3.4.5</mybatis.version>
<!--mysql-->
<mysql.version>5.1.44</mysql.version>
<!--pagehelper分页jar依赖-->
<pagehelper.version>5.1.2</pagehelper.version>
<!--mybatis与spring集成jar依赖-->
<mybatis.spring.version>1.3.1</mybatis.spring.version>
<!--3.dbcp2连接池相关 druid-->
<commons.dbcp2.version>2.1.1</commons.dbcp2.version>
<commons.pool2.version>2.4.3</commons.pool2.version>
<!--4.log日志相关-->
<log4j2.version>2.9.1</log4j2.version>
<!--5.其他-->
<junit.version>4.12</junit.version>
<servlet.version>4.0.0</servlet.version>
<lombok.version>1.18.2</lombok.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>
<!--2.mybatis相关-->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>${mybatis.version}</version>
</dependency>
<!--mysql-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql.version}</version>
</dependency>
<!--pagehelper分页插件jar包依赖-->
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper</artifactId>
<version>${pagehelper.version}</version>
</dependency>
<!--mybatis与spring集成jar包依赖-->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>${mybatis.spring.version}</version>
</dependency>
<!--3.dbcp2连接池相关-->
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-dbcp2</artifactId>
<version>${commons.dbcp2.version}</version>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-pool2</artifactId>
<version>${commons.pool2.version}</version>
</dependency>
<!--4.log日志相关依赖-->
<!--核心log4j2jar包-->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
<version>${log4j2.version}</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
<version>${log4j2.version}</version>
</dependency>
<!--web工程需要包含log4j-web,非web工程不需要-->
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-web</artifactId>
<version>${log4j2.version}</version>
</dependency>
<!--5.其他-->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>${junit.version}</version>
<!-- <scope>test</scope>-->
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>${servlet.version}</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>${lombok.version}</version>
<scope>provided</scope>
</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>index</finalName>
<resources>
<!--解决mybatis-generator-maven-plugin运行时没有将XxxMapper.xml文件放入target文件夹的问题-->
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.xml</include>
</includes>
</resource>
<!--解决mybatis-generator-maven-plugin运行时没有将jdbc.properites文件放入target文件夹的问题-->
<resource>
<directory>src/main/resources</directory>
<includes>
<include>jdbc.properties</include>
<include>*.xml</include>
</includes>
</resource>
</resources>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>${maven.compiler.plugin.version}</version>
<configuration>
<source>${maven.compiler.source}</source>
<target>${maven.compiler.target}</target>
<encoding>${project.build.sourceEncoding}</encoding>
</configuration>
</plugin>
<plugin>
<groupId>org.mybatis.generator</groupId>
<artifactId>mybatis-generator-maven-plugin</artifactId>
<version>1.3.2</version>
<dependencies>
<!--使用Mybatis-generator插件不能使用太高版本的mysql驱动 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql.version}</version>
</dependency>
</dependencies>
<configuration>
<overwrite>true</overwrite>
</configuration>
</plugin>
<plugin>
<artifactId>maven-clean-plugin</artifactId>
<version>3.1.0</version>
</plugin>
<!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging -->
<plugin>
<artifactId>maven-resources-plugin</artifactId>
<version>3.0.2</version>
</plugin>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.8.0</version>
</plugin>
<plugin>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.22.1</version>
</plugin>
<plugin>
<artifactId>maven-war-plugin</artifactId>
<version>3.2.2</version>
</plugin>
<plugin>
<artifactId>maven-install-plugin</artifactId>
<version>2.5.2</version>
</plugin>
<plugin>
<artifactId>maven-deploy-plugin</artifactId>
<version>2.8.2</version>
</plugin>
</plugins>
</build>
</project>
缺少jstl+standard的这两个jar包会报java.lang.NoClassDefFoundError: javax/servlet/jsp/jstl/core/Config,那是因为org.springframework.web.servlet.view.JstlView在视图解析时需要这二个jar包。
创建 spring-mvc.xml,这里又在spring-mvc.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 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.CloudJun及子子孙孙包下的控制器(扫描范围过大,耗时)--> <context:component-scan base-package="com.CloudJun"/> <!--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/**"/>--> <!-- 处理static包里的所有静态资源 --> <mvc:resources location="/static/" mapping="/static/**"/> </beans>
配置web.xml
配置步骤:
配置Spring与Web项目集成
配置中文乱码过滤器
配置SpringMVC核心控制器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_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>
创建一个 IndexController
package com.CloudJun.web;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
/**
* @author CloudJun
* @create 2023-09-04 16:19
*/
@Controller
@RequestMapping("/Cloud")
public class IndexController {
@RequestMapping("/Jun")
public String primary(){
System.out.println("SpringMVC 使用。。。。。");
return "hello";
}
}
之后我们开启服务器,在浏览器中发送这个请求测试 : http://localhost:8080/index/Cloud/Jun
( 根据自己的配置进行请求访问才可以 )
测试结果 :
( 2 ) 静态资源处理
首先先将随便一个静态资源放进项目的静态资源包中
这里我们以以下图片为例
然后在 spring-mvc.xml 配置文件中增加以下代码
<!-- 处理static包里的所有静态资源 --> <mvc:resources location="/static/" mapping="/static/**"/>
( 这里是会处理所有的静态资源,也就是处理static包里的所有静态资源 )
开启服务器,在浏览器中发送这个请求测试,请求根据所在的资源目录结构来输入。
如我的图片资源目录结构如图 :
所有我需要发送的请求为 : http://localhost:8080/index/static/images/preview.jpg
测试结果如图
给我们带来的收获
学习了SpringMVC的工作流程和静态资源处理,我们可以获得以下收获:
- 1. 理解了SpringMVC的整体工作流程:学习了请求的流程,包括DispatcherServlet的作用、HandlerMapping的匹配过程、HandlerAdapter的处理过程以及ViewResolver的视图解析过程等。这使我们能够更好地理解SpringMVC框架的工作原理,从而更好地进行开发和调试。
- 2. 掌握了静态资源处理的方法:学习了如何配置SpringMVC来处理静态资源,包括图片、样式表、脚本等。这使我们可以将静态资源与动态内容分离,提高网站的性能和可维护性。
- 3. 学会了如何配置静态资源的映射:学习了如何配置SpringMVC来映射静态资源的URL路径,以及如何配置静态资源的存放位置。这使我们可以自定义静态资源的URL路径,更好地组织和管理静态资源。
- 4. 了解了静态资源的缓存机制:学习了如何配置SpringMVC来启用静态资源的缓存,以减少网络请求和提高网站的加载速度。这使我们可以更好地优化网站的性能,提升用户体验。
总之,学习了SpringMVC的工作流程和静态资源处理,我们可以更好地理解和使用SpringMVC框架,提高开发效率和网站性能。