网站首页 > 技术文章 正文
Servlet和Spring MVC都是在Java Web开发中用于处理Web请求的技术,其中Servlet是Java EE规范的一部分,而Spring MVC是Spring Framework的一部分,属于Spring生态系统的一部分。
接下来,我们就来详细的介绍一下二者之间的区别,并且结合底层原理对其进行说明。
Servlet是什么?
Servlet 是 Java EE(现在叫 Jakarta EE)中最核心的组件之一,它是一个接口,主要用于处理HTTP请求。通过实现 javax.servlet.Servlet 接口,来处理客户端发送的请求并生成响应。
其工作流程如下。
- 第一步、 客户端一般情况下这个客户端通常是指浏览器,发起一个HTTP请求,Servlet会接收到该请求。
- 第二步、当请求到达Servlet容器(例如Tomcat)之后,Servlet容器会解析请求的URL,并根据映射规则找到相应的Servlet。
- 第三步、如果Servlet还没有被加载,容器会先实例化它。每个请求会触发Servlet的 service() 方法,该方法会根据HTTP请求类型(GET、POST等)调用相应的doGet()或doPost()等方法。
- 第四步、Servlet处理完请求后,会将生成的响应数据(通常是HTML或JSON等)返回给客户端。
- 第五步、当服务器关闭或Servlet不再需要时,容器会调用destroy()方法来销毁Servlet实例。
这也对应了之前的分享中,我们介绍的关于过滤器的操作内容,如下所示。
@WebServlet("/hello")
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException {
resp.setContentType("text/html");
PrintWriter out = resp.getWriter();
out.println("<h1>Hello, Servlet!</h1>");
}
}
Servlet要求开发者可以手动处理HTTP请求的各个方面,包括从HttpServletRequest中获取参数,来生成HttpServletResponse响应,以及处理HTTP方法(如GET、POST等)。当然也可以通过转发请求给JSP引擎来渲染动态HTML页面。在默认情况下,Servlet是单实例的,也就是说每个Servlet实例可以处理多个请求,因此需要开发者保证线程安全。
Spring MVC是什么?
Spring MVC是Spring Framework框架提供的一个基于Servlet实现的Web框架,它遵循MVC(Model-View-Controller)设计模式,其设计的目的就是为了简化Web应用的开发。并且Spring MVC是建立在Servlet API之上,与Servlet的不同就是它提供了更高层次的抽象,并对复杂的Web请求处理提供了更强大的支持。
Spring MVC的工作流程包括多个组件,它通过DispatcherServlet作为前端控制器来协调请求处理,其处理流程如下所示。
- 第一步、和Servlet一样,客户端(浏览器)发送HTTP请求。
- 第二步、请求首先到达Spring MVC的核心组件DispatcherServlet其本质上也是一个Servlet。
- 第三步、这个时候DispatcherServlet会使用HandlerMapping来根据URL找到对应的处理器,这个处理器通常是一个控制器Controller。
- 第四步、DispatcherServlet会调用映射到的控制器中的方法。而控制器方法通常会返回一个 ModelAndView对象这个对象包含视图名和模型数据,用来进行View层的数据渲染展示。
- 第五步、DispatcherServlet通过ViewResolver来将返回的视图名解析为具体的视图,例如JSP或Thymeleaf模板引擎。
- 第六步、视图负责渲染最终的HTML或其他格式的响应内容,最后响应返回给客户端。
如下所示,是一个简单的SpringMVC的示例。
@Controller
public class HelloController {
@GetMapping("/hello")
public String sayHello(Model model) {
model.addAttribute("message", "Hello, Spring MVC!");
return "hello"; // 返回视图名 "hello"
}
}
Spring MVC主要依靠注解来定义控制器和映射规则,简化了开发。例如我们比较常见的注解有@Controller, @RequestMapping, @GetMapping, @PostMapping等,Spring MVC会自动将HTTP请求中的参数绑定到控制器方法的参数中,不需要手动从 HttpServletRequest 获取。提供了众多拦截器(interceptors)、过滤器(filters)等机制,可以方便地进行功能扩展。
通过内置的视图解析机制,可以轻松集成JSP、Thymeleaf等视图技术。支持对表单数据的自动绑定、表单验证和错误处理。
底层差异详解
控制器的实现方式
- Servlet直接实现或继承HttpServlet类来实现的具体操作,我们需要通过重写doGet()、doPost()等方法来处理HTTP请求操作。
- Spring MVC控制器不需要继承特定类,它通常使用@Controller注解来标记控制类类,然后通过使用@RequestMapping或其他映射注解来定义URL的映射关系。这种方式避免了和具体的Servlet API绑定,使得Spring MVC的使用更加的灵活。
请求处理模型
- Servlet是单个类对应单个URL映射,处理过程较为手动。需要开发者去处理大量的处理请求参数、构建响应。
- Spring MVC采用了DispatcherServlet作为中央控制器,将所有的请求经过它来进行处理,然后通过各种策略模式的组件,例如HandlerMapping, HandlerAdapter, ViewResolver等进行分发和处理。这样对于开发者来讲,只需要专注于业务逻辑。而底层的许多重复工作,例如请求参数的处理、视图解析等则是由Spring框架自动完成。
对象和数据管理
- Servlet处理请求的生命周期较短,主要通过HttpServletRequest和HttpServletResponse传递请求和响应的数据。Servlet则本身只专注于请求和响应的交互,对于状态的管理则需要通过会话Session或者通过上下文Context机制来实现。
- Spring MVC中的数据传递更加高级,例如通过Model或ModelAndView来传递数据到视图。Spring MVC还整合了@ModelAttribute等注解来处理表单提交的数据绑定和数据校验,减少了手动从请求中获取参数的工作。
扩展性和模块化
- Servlet本身比较底层,虽然灵活,但很多功能需要手动实现,比如会话管理、跨站请求伪造(CSRF)防护等,开发者必须依赖外部库或者自己实现。
- Spring MVC提供了大量现成的组件,诸如数据绑定、校验、异常处理、国际化、模板引擎集成等。开发者可以通过扩展HandlerInterceptor、 ControllerAdvice 等类进行扩展,也可以利用Spring生态的众多功能(如Spring Security、Spring Data等)来增强Web应用。
依赖注入和AOP
- Servlet本身不支持依赖注入,也没有AOP能力。需要结合Java EE中的CDI(Context and Dependency Injection)或手动管理依赖。
- Spring MVC完全依赖Spring的核心功能即IOC容器和AOP,控制器类的依赖可以通过Spring的自动注入例如 @Autowired来管理,极大简化了组件之间的依赖管理。AOP(Aspect-Oriented Programming)也可以用于控制器中,以处理日志、事务等横切关注点。
总结
Servlet是Java Web开发的底层技术,它是Java EE的一部分,要求开发者手动处理请求、响应、参数等内容。Servlet的API较为原始,虽然功能强大,但并没有提供高级的Web框架功能。Spring MVC是基于Servlet的一个Web框架,它简化了Web开发,通过 DispatcherServlet 提供了更加模块化、可扩展的方式来处理Web请求。Spring MVC 依赖Spring框架的依赖注入和其他组件,极大提升了开发效率,并且集成了大量Web开发中常用的功能。
两者的区别可以类比于“手工打造”与“利用框架搭建”,Spring MVC在Servlet的基础上进行了高度封装,简化了开发流程并增强了扩展性。
猜你喜欢
- 2024-12-25 Spring Boot整合Spring Cloud GateWay代理第三方应用的调用接口?
- 2024-12-25 Java 近期新闻:Hibernate 6.0、JobRunr 5.0、JHipster 7.8.0
- 2024-12-25 Keycloak Servlet Filter Adapter使用
- 2024-12-25 如何在Spring Boot中保证RESTful接口的安全性?
- 2024-12-25 Java项目实战第6天:登录业务的实现
- 2024-12-25 JavaEE概述总结:Servlet生命周期+JSP内置对象
- 2024-12-25 SpringBoot 无感刷新 Token springboot的token
- 2024-12-25 若依开发框架解析笔记(7)-jwt的应用
- 2024-12-25 Spring MVC中提供了哪些扩展机制?如何使用这些扩展机制?
- 2024-12-25 49个Spring经典面试题总结(附带答案)
- 02-21走进git时代, 你该怎么玩?_gits
- 02-21GitHub是什么?它可不仅仅是云中的Git版本控制器
- 02-21Git常用操作总结_git基本用法
- 02-21为什么互联网巨头使用Git而放弃SVN?(含核心命令与原理)
- 02-21Git 高级用法,喜欢就拿去用_git基本用法
- 02-21Git常用命令和Git团队使用规范指南
- 02-21总结几个常用的Git命令的使用方法
- 02-21Git工作原理和常用指令_git原理详解
- 最近发表
- 标签列表
-
- cmd/c (57)
- c++中::是什么意思 (57)
- sqlset (59)
- ps可以打开pdf格式吗 (58)
- phprequire_once (61)
- localstorage.removeitem (74)
- routermode (59)
- vector线程安全吗 (70)
- & (66)
- java (73)
- org.redisson (64)
- log.warn (60)
- cannotinstantiatethetype (62)
- js数组插入 (83)
- resttemplateokhttp (59)
- gormwherein (64)
- linux删除一个文件夹 (65)
- mac安装java (72)
- reader.onload (61)
- outofmemoryerror是什么意思 (64)
- flask文件上传 (63)
- eacces (67)
- 查看mysql是否启动 (70)
- java是值传递还是引用传递 (58)
- 无效的列索引 (74)