SpringMVC实战:动态时钟
引言
在现代 Web 开发中,选择一个合适的框架对于项目的成功至关重要。Spring MVC 作为 Spring 框架的核心模块之一,以其清晰的架构、强大的功能和高度的可配置性,成为了 Java Web 开发领域的主流选择。本文将通过一个“动态时钟”的实战项目,深入分析 Spring MVC 模式,并分享在开发过程中遇到的问题及解决方案。
Spring MVC 模式分析
什么是 Spring MVC?
Spring MVC 是一个基于 Java 的 Web 框架,它实现了著名的 模型-视图-控制器 (Model-View-Controller, MVC) 设计模式。它旨在分离应用程序的不同职责,使得开发人员能够更清晰地组织代码,提高可维护性和可扩展性。
MVC 模式的核心概念
在深入 Spring MVC 之前,我们先回顾一下 MVC 模式的三个核心组件:
-
Model (模型): 负责封装应用程序的数据和业务逻辑。它独立于用户界面,处理数据的存储、检索、更新和验证。在 Web 应用中,模型通常是 Java Bean 或 POJO (Plain Old Java Object)。
-
View (视图): 负责数据的展示,即用户界面。它从模型中获取数据,并以用户友好的方式呈现出来。视图通常不包含任何业务逻辑,只负责显示。在 Java Web 中,JSP (JavaServer Pages)、Thymeleaf、FreeMarker 等都可以作为视图技术。
-
Controller (控制器): 充当模型和视图之间的协调者。它接收用户的输入(通常是 HTTP 请求),处理这些输入,调用模型来执行相应的业务逻辑,然后选择合适的视图来显示结果。控制器本身不处理业务逻辑,它只是将请求转发给模型,并将模型的结果传递给视图。
Spring MVC 在 MVC 模式中的实现
Spring MVC 对 MVC 模式进行了精巧的实现,其核心是一个名为 DispatcherServlet
的前端控制器。以下是 Spring MVC 请求处理的典型流程:
-
DispatcherServlet
(前端控制器):-
作为整个 Spring MVC 应用程序的入口点,它拦截所有(或特定模式的)进来的 HTTP 请求。
-
它的作用类似于一个总调度员,将请求分发给合适的处理器。
-
-
Handler Mapping (处理器映射):
-
DispatcherServlet
接收到请求后,会查询HandlerMapping
。 -
HandlerMapping
的职责是根据请求的 URL 路径,找到能够处理该请求的 处理器 (Handler),通常是带有@Controller
和@RequestMapping
注解的控制器方法。
-
-
Controller (控制器):
-
找到对应的控制器方法后,
DispatcherServlet
会调用该方法。 -
控制器方法执行业务逻辑(可能通过调用服务层或数据访问层),处理请求参数,并准备需要展示给视图的数据。
-
控制器通常返回一个
ModelAndView
对象或一个逻辑视图名,其中包含模型数据和视图信息。
-
-
ModelAndView
(模型和视图):-
这是一个封装了模型数据和逻辑视图名的对象。模型数据是键值对形式,视图名是字符串,用于标识要渲染的视图。
-
-
View Resolver (视图解析器):
-
DispatcherServlet
接收到控制器返回的逻辑视图名后,会将其交给ViewResolver
。 -
ViewResolver
的职责是将逻辑视图名解析为实际的视图资源(例如,一个 JSP 文件的路径)。在我们的项目中,InternalResourceViewResolver
会将 "clock" 解析为/WEB-INF/views/clock.jsp
。
-
-
View (视图):
-
一旦
ViewResolver
找到实际的视图资源,DispatcherServlet
就会将模型数据传递给该视图。 -
视图负责渲染模型数据,生成最终的 HTML、XML 或其他格式的响应,并将其发送回客户端浏览器。
-
Spring MVC 的优点
-
职责分离清晰: 模型、视图、控制器各司其职,代码结构清晰,易于理解和维护。
-
灵活性和可配置性: Spring MVC 提供了大量的可插拔组件(如
HandlerMapping
、ViewResolver
),允许开发人员根据需求进行高度定制。 -
易于测试: 由于各层职责分离,控制器可以独立于视图和模型进行单元测试,提高了测试效率。
-
强大的生态系统: 作为 Spring 框架的一部分,Spring MVC 可以无缝集成 Spring 的其他模块,如 Spring Security、Spring Data 等,提供全面的企业级解决方案。
-
RESTful 支持: 内置对 RESTful Web 服务的强大支持,使得构建 API 变得简单。
项目实战:动态时钟
项目目标
我们的目标是构建一个基于 Spring MVC 的 Web 应用程序,其中包含一个动态变化的模拟时钟。时钟的画面将通过 JSP 页面呈现,并利用前端技术(HTML、CSS、JavaScript)实现其动态效果和居中显示。
技术栈
-
后端框架: Spring MVC
-
视图技术: JSP (JavaServer Pages)
-
前端样式: Tailwind CSS (通过 CDN 引入)
-
前端逻辑: JavaScript (实现指针动态旋转)
-
构建工具: Maven
-
应用服务器: Apache Tomcat
项目结构概览
一个典型的 Spring MVC Web 项目结构如下:
your-clock-project/
├── src/
│ ├── main/
│ │ ├── java/
│ │ │ └── com/
│ │ │ └── example/
│ │ │ └── controller/
│ │ │ └── ClockController.java <-- Spring MVC 控制器
│ │ ├── resources/
│ │ │ └── spring-mvc-config.xml <-- Spring MVC 配置文件
│ │ └── webapp/
│ │ ├── WEB-INF/
│ │ │ ├── web.xml <-- Web 应用部署描述符
│ │ │ └── views/
│ │ │ └── clock.jsp <-- JSP 视图页面
│ │ └── ...
├── pom.xml <-- Maven 项目对象模型文件
关键文件说明:
-
pom.xml
: Maven 项目配置文件,管理项目依赖(Spring MVC、JSP API、Servlet API 等)和构建插件(如maven-war-plugin
和tomcat8-maven-plugin
)。我们通过<finalName>web10</finalName>
将 WAR 包名称设置为web10.war
。 -
web.xml
: Web 应用程序部署描述符,配置DispatcherServlet
作为前端控制器,并指定其 Spring MVC 配置文件的位置。 -
spring-mvc-config.xml
: Spring MVC 配置文件,启用注解驱动的 MVC,配置组件扫描(扫描控制器),并定义InternalResourceViewResolver
来解析 JSP 视图。 -
ClockController.java
: Spring MVC 控制器,使用@RequestMapping("/clock")
将 HTTP GET 请求映射到showClockPage()
方法,该方法返回逻辑视图名"clock"
。 -
clock.jsp
: JSP 视图页面,包含动态时钟的 HTML 结构、Tailwind CSS 样式和 JavaScript 逻辑。前端 JavaScript 负责获取当前时间并实时更新时针、分针、秒针的旋转角度,实现动态效果。
部署与运行
-
构建 WAR 包: 在项目根目录运行
mvn clean install
命令,将在target/
目录下生成web10.war
文件。 -
部署到 Tomcat:
-
手动部署: 将
web10.war
复制到 Tomcat 安装目录下的webapps
文件夹。 -
使用 Maven 插件: 在
pom.xml
中配置tomcat8-maven-plugin
并设置settings.xml
中的 Tomcat 管理员凭据后,运行mvn tomcat8:deploy
(部署到外部 Tomcat) 或mvn tomcat8:run
(在嵌入式 Tomcat 中运行)。
-
-
访问应用: 部署成功后,在浏览器中访问
http://localhost:8080/web10/clock
即可看到动态时钟。
项目效果截图

动态时钟效果截图
遇到的问题与解决方案
在项目开发和部署过程中,我们遇到了一些常见的问题:
-
Missing artifact jakarta.servlet.jsp:jakarta.servlet.jsp-api:jar:2.3.3
:-
问题: Maven 无法找到或下载
jakarta.servlet.jsp-api
的2.3.3
版本依赖。这通常是由于本地仓库缓存问题或远程仓库同步延迟导致。 -
解决方案:
-
首先尝试运行
mvn clean install -U
强制更新依赖。 -
如果无效,将
pom.xml
中jakarta.servlet.jsp.version
的版本号更新到更稳定和常用的版本,例如3.0.0
。
-
-
-
ClassFormatException: Invalid byte tag in constant pool: 19
(Tomcat 版本与依赖不兼容):-
问题: 当使用
tomcat7-maven-plugin
(对应 Tomcat 7) 运行项目时,Tomcat 无法处理jakarta.xml.bind-api
等较新 Jakarta EE 依赖中包含的module-info.class
文件。这表明 Tomcat 7 的内部类加载器与这些新特性不兼容。 -
解决方案: 将
pom.xml
中的tomcat7-maven-plugin
替换为tomcat8-maven-plugin
并更新到兼容的版本(例如3.2.2
)。Tomcat 8.5+ 对 Java 8 和 Jakarta EE 8 的兼容性更好。
-
-
HTTP Status 404 - Not Found
(直接访问WEB-INF
目录下的 JSP):-
问题: 尝试直接在浏览器中访问
http://localhost:8080/web10/WEB-INF/views/clock.jsp
时,Tomcat 返回 404 错误。 -
解决方案: 这是 Java Web 应用程序的预期安全行为。
WEB-INF
目录是受保护的,其中的资源不能被客户端直接访问。正确的访问方式是通过 Spring MVC 控制器映射的 URL,即http://localhost:8080/web10/clock
。DispatcherServlet
会在服务器内部将请求转发到WEB-INF
下的 JSP 视图。
-
总结
通过本次动态时钟 Web 应用的开发,我不仅实践了 Spring MVC 框架的基本配置和使用,更深入理解了其背后的 MVC 设计模式原理。从请求的拦截、分发到视图的渲染,Spring MVC 提供了一套清晰且高效的机制来构建可维护、可扩展的 Web 应用程序。同时,解决实际开发中遇到的依赖冲突和部署问题,也进一步加深了对 Maven、Tomcat 和 Java Web 规范的理解。