Spring Boot的实用内置功能详解
Spring Boot作为一款备受欢迎的Java框架,以其简洁、高效和易用的特点,赢得了广大开发者的青睐。其内置的多种功能更是为开发者提供了极大的便利,本文将详细介绍Spring Boot中记录请求数据、请求/响应包装器、特殊的过滤器Filter以及Controller接口定义等实用内置功能。
一、记录请求数据
在Spring Boot中,记录请求数据是一项非常重要的功能,它有助于开发者在调试和监控应用程序时,获取详尽的请求信息。Spring Boot提供了多种方式来记录请求数据,其中利用AOP(面向切面编程)和过滤器Filter是比较常见的两种方式。
-
利用AOP记录请求数据
AOP允许开发者在方法执行的前后添加额外的逻辑,而不需要修改方法本身的代码。在Spring Boot中,可以利用AOP切面来拦截Controller层的方法,从而记录请求数据。
定义一个切面类,该类包含前置通知(在方法执行前记录请求数据)和后置通知(在方法执行后记录响应数据)。通过注解
@Aspect和@Component将该切面类注册为Spring容器中的一个Bean。在切面类中,使用@Pointcut注解定义切入点表达式,用于匹配需要拦截的方法。然后,使用@Before和@AfterReturning注解分别定义前置通知和后置通知的逻辑。 -
利用过滤器Filter记录请求数据
过滤器Filter是Servlet规范中的一部分,它允许开发者在请求到达Servlet之前或响应发送给客户端之前对请求和响应进行预处理和后处理。在Spring Boot中,可以通过实现
javax.servlet.Filter接口来创建自定义的过滤器。在过滤器类中,重写
doFilter方法,在该方法中实现日志记录逻辑。可以从ServletRequest和ServletResponse对象中获取请求和响应的详细信息,如请求URL、请求参数、请求头、响应状态码等。然后,使用日志框架(如Logback或Log4j)将这些信息记录到日志文件中。此外,Spring Boot还提供了
AbstractRequestLoggingFilter和CommonsRequestLoggingFilter等内置过滤器,用于记录请求数据。这些过滤器可以通过简单的配置即可使用,无需编写额外的代码。
二、请求/响应包装器
请求/响应包装器是Spring Boot另一个强大的功能,它允许开发者在请求处理过程中增强HttpServletRequest和HttpServletResponse的功能。通过包装器,开发者可以在不修改原始请求和响应对象的情况下,添加额外的逻辑或信息。
-
请求包装器
使用
ContentCachingRequestWrapper可以对请求内容进行缓存,使得同一请求可以多次读取。这在记录日志或数据修改时非常有用。例如,当需要记录请求体中的JSON数据时,可以使用请求包装器来缓存请求体,并在日志记录完成后继续处理请求。 -
响应包装器
使用
ContentCachingResponseWrapper可以对响应内容进行缓存和修改。通过包装器,开发者可以在最终响应之前修改响应头、响应体或添加额外的信息。这对于实现响应的压缩、加密或添加自定义的响应头等功能非常有帮助。
三、特殊的过滤器Filter
除了上述提到的记录请求数据的过滤器外,Spring Boot还支持多种特殊的过滤器,用于实现不同的功能。
-
OncePerRequestFilter
OncePerRequestFilter是Spring框架中的一个重要过滤器基类,它确保过滤器逻辑在请求的生命周期中只被执行一次。无论请求如何转发或包含,这种机制都能极大地减少重复处理的风险,确保请求数据处理的高效性。 -
AbstractPreAuthenticatedProcessingFilter
AbstractPreAuthenticatedProcessingFilter是一个用于预认证处理的过滤器基类。它允许开发者在请求到达应用程序之前,通过某种方式(如HTTP头、SSL会话等)进行用户的预认证。这对于实现基于令牌的身份验证或单点登录等功能非常有用。 -
其他自定义过滤器
开发者还可以根据自己的需求创建自定义的过滤器。通过实现
javax.servlet.Filter接口或继承Spring Boot提供的过滤器基类(如OncePerRequestFilter),可以轻松地创建具有特定功能的过滤器。
四、Controller接口定义
在Spring Boot中,Controller层负责处理HTTP请求并返回响应。Spring Boot提供了多种方式来定义Controller接口,以满足不同的需求。
-
使用
@RestController注解@RestController是一个组合注解,它结合了@Controller和@ResponseBody的功能。使用@RestController注解的类中的所有方法都会默认返回JSON或XML格式的响应体。这对于实现RESTful API非常有用。 -
使用
@RequestMapping注解@RequestMapping注解用于将HTTP请求映射到特定的处理方法上。它可以定义在类或方法级别上,用于匹配请求的URL、请求方法(如GET、POST等)、请求参数等。通过@RequestMapping注解,开发者可以灵活地定义Controller接口的处理逻辑。 -
使用
@GetMapping、@PostMapping等衍生注解Spring Boot还提供了
@GetMapping、@PostMapping、@PutMapping、@DeleteMapping等衍生注解,用于简化@RequestMapping注解的配置。这些衍生注解分别用于匹配特定的HTTP请求方法,使得代码更加简洁易读。 -
使用
@RequestBody和@ResponseBody注解@RequestBody注解用于将HTTP请求体中的内容绑定到方法参数上,通常用于处理POST请求中的JSON或XML数据。@ResponseBody注解用于将方法的返回值直接写入HTTP响应体中,通常用于返回JSON或XML格式的响应数据。
综上所述,Spring Boot的内置功能为开发者提供了极大的便利和灵活性。通过记录请求数据、使用请求/响应包装器、配置特殊的过滤器以及灵活定义Controller接口等方式,开发者可以轻松地构建高效、可靠和易于维护的Web应用程序。
五、例子
1. 日志记录
Spring Boot提供了一个内置的日志记录解决方案,通过AbstractRequestLoggingFilter可以记录请求的详细信息。例如,使用CommonsRequestLoggingFilter来记录请求的参数、请求体、请求头和客户端信息:
@Configuration
public class RequestLoggingConfig { @Bean public CommonsRequestLoggingFilter logFilter() { CommonsRequestLoggingFilter filter = new CommonsRequestLoggingFilter(); filter.setIncludeQueryString(true); filter.setIncludePayload(true); filter.setIncludeHeaders(true); filter.setIncludeClientInfo(true); filter.setAfterMessagePrefix("REQUEST DATA-"); return filter; }
}
然后,需要配置日志级别为DEBUG,以详细记录请求信息:
logging.level.org.springframework.web.filter.CommonsRequestLoggingFilter=DEBUG
2. 过滤器(Filter)
过滤器可以用于对请求和响应进行预处理和后处理。例如,可以创建一个自定义过滤器来记录请求的URL:
package com.example.filter; import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException; public class MyFilter implements Filter { @Override public void init(javax.servlet.FilterConfig filterConfig) throws ServletException { // 过滤器初始化逻辑 } @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { if (servletRequest instanceof HttpServletRequest) { System.out.println("URL: " + ((HttpServletRequest) servletRequest).getRequestURL()); } // 调用过滤器链中的下一个过滤器 filterChain.doFilter(servletRequest, servletResponse); } @Override public void destroy() { // 过滤器销毁逻辑 }
}
然后,使用FilterRegistrationBean来注册这个自定义过滤器:
@Configuration
public class FilterConfig { @Bean public FilterRegistrationBean registrationBean() { FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(new MyFilter()); filterRegistrationBean.addUrlPatterns("/*"); return filterRegistrationBean; }
}
3. 控制器(Controller)定义
在Spring Boot中,控制器通常使用@RestController注解来定义。这个注解是@Controller和@ResponseBody的组合,表示该类可以处理HTTP请求,并且返回值会自动序列化为JSON格式。
例如,定义一个简单的用户控制器:
@RestController
@RequestMapping("/users")
public class UserController { @GetMapping("/{id}") public User getUserById(@PathVariable Long id) { // 假设这里有一个服务层来获取用户信息 return new User(id, "用户名" + id); } @PostMapping public ResponseEntity<User> createUser(@RequestBody User user) { // 假设这里有一个服务层来创建用户 return ResponseEntity.ok(user); }
}
其中,User类是一个简单的POJO类,包含用户的ID和用户名等信息。
4. 请求和响应处理
Spring Boot提供了请求和响应包装器(HttpServletRequestWrapper和HttpServletResponseWrapper),用于增强原生HttpServletRequest和HttpServletResponse对象的功能。例如,可以使用ContentCachingRequestWrapper来缓存请求体,以便多次读取:
@Component
public class RequestWrapperFilter extends OncePerRequestFilter { @Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException { ContentCachingRequestWrapper requestWrapper = new ContentCachingRequestWrapper(request); // 可以在这里处理请求数据 byte[] body = requestWrapper.getContentAsByteArray(); // ... filterChain.doFilter(requestWrapper, response); }
}
同样地,可以使用ContentCachingResponseWrapper来缓存响应体,以便在响应提交给客户端之前进行修改:
@Component
public class ResponseWrapperFilter extends OncePerRequestFilter { @Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException { ContentCachingResponseWrapper responseWrapper = new ContentCachingResponseWrapper(response); filterChain.doFilter(request, responseWrapper); // 可以在这里处理响应数据 byte[] body = responseWrapper.getContentAsByteArray(); // 处理body,例如添加签名 responseWrapper.setHeader("X-Signature", "some-signature"); // 必须调用此方法以将响应数据发送到客户端 responseWrapper.copyBodyToResponse(); }
}
这些例子展示了Spring Boot中的一些关键功能和概念。通过实践这些例子,您可以更好地理解Spring Boot的工作原理,并为您的项目开发提供有用的参考。
相关文章:
Spring Boot的实用内置功能详解
Spring Boot作为一款备受欢迎的Java框架,以其简洁、高效和易用的特点,赢得了广大开发者的青睐。其内置的多种功能更是为开发者提供了极大的便利,本文将详细介绍Spring Boot中记录请求数据、请求/响应包装器、特殊的过滤器Filter以及Controlle…...
撸猫变梳毛?怎么解决猫咪掉毛问题?好用的宠物空气净化器推荐
秋风一吹,新一轮的猫咪换毛季又到了,这也意味着我失去了撸猫自由。我每天的治愈方式就是下班撸猫,抚摸着柔软的毛发,好像一天的烦恼都消除了。可是一到换毛季,猫还没撸两下,先从猫咪身上带下一手毛…...
人声分离免费软件,六款好用软件处理音乐更轻松!
在这个数字化音乐时代,无论是专业音乐人还是音乐爱好者,都渴望在创作与编辑过程中拥有更多便捷高效的工具。人声分离技术,作为音乐后期制作中的一项关键技术,能够精准地将歌曲中的人声与伴奏分离,极大地拓宽了音乐创作…...
数据分析Power BI设置万为单位的数据
玩过Power BI的同学都知道,power BI在度量值设置单位里,唯独没有万这个单位,但是我们可以自定义,操作过程如下: 1.用DAX新建单位表 单位 SELECTCOLUMNS( { ( "元", 1), ("万",10000), ("千…...
(AI 生成) 新时代游击方式: 利用 “灵活就业“ 红利
注意: 本文内容为 AI 大模型生成, 仅供参考. 提示词: 写一篇短文, 500 字左右, 标题为: 新时代游击方式: 利用 “灵活就业” 红利 1 豆包 《新时代游击方式:利用“灵活就业”红利》 在新时代的大舞台上,“灵活就业”犹如一块熠熠生辉的宝藏,…...
Unity UndoRedo(撤销重做)功能
需求 撤销与重做功能 思考 关于记录的数据的两点思考: 记录操作记录影响显示和逻辑的所有数据 很显然这里就要考虑取舍了: 记录操作 这种方案只需要记录每一步的操作,具体这个操作要怎么渲染和实现出来完全需要自己去实现,这…...
28条有关人工智能的名言
当谈到人工智能(AI)的潜力和潜在风险,以及无人类干预的机器学习和推理过程时,目前尚存在许多不同的观点。 只有时间会告诉我们,这些语录中哪一条是最接近未来的真实情况的。在我们尚未到达目的地之前,想一想…...
搞机器视觉项目看不起搞机器视觉培训的,实际上怎么样
搞机器视觉项目第一要务就是验收回款,往往欠款的非常严重,多数还要打通人际关系需要大量的成本。大多数机器视觉检测项目具有一定的风险,客户要求不明确,技术评估不充分,往往伴随着失败的可能性。所以做项目又累又担风…...
使用Jenkins部署项目
部署中的痛点 为什么要用Jenkins?我说下我以前开发的痛点,在一些中小型企业,每次开发一个项目完成后,需要打包部署,可能没有专门的运维人员,只能开发人员去把项目打成一个exe包,可能这个项目已…...
【机器学习与神经网络荣获诺贝尔奖】科学边界的扩展及技术革新
【机器学习与神经网络荣获诺贝尔奖】科学边界的扩展及技术革新 1)科学交叉融合的体现2)方法论的创新3)社会影响的考量 一、机器学习与神经网络的发展前景1)生产制造领域2)金融领域3)医疗领域 二、机器学习和…...
Javascript扩展符号(...)使用说明
在 ES6 中,扩展运算符(spread operator)... 可以用于在函数调用、数组字面量或对象字面量中展开数组或对象。以下是扩展运算符的一些常见用法: 1. 在函数调用中使用扩展运算符 扩展运算符可以在函数调用时展开数组或对象&#x…...
giugughk
c语言中的小小白-CSDN博客c语言中的小小白关注算法,c,c语言,贪心算法,链表,mysql,动态规划,后端,线性回归,数据结构,排序算法领域.https://blog.csdn.net/bhbcdxb123?spm1001.2014.3001.5343 给大家分享一句我很喜欢我话: 知不足而奋进,望远山而前行&am…...
【微服务】网关 - Gateway(下)(day8)
网关过滤工厂 在上一篇文章中,主要是对网关进行了一个总体的介绍,然后对网关中的断言进行了一个描述。在这篇文章中,主要是对网关中的最后一大核心——过滤进行介绍。 当客户端发送过来的请求经过断言之后,如果还想在请求前后添…...
【C#】创建一个控制台应用程序来管理学生成绩
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 在C#中创建一个控制台应用程序来管理学生成绩编写程序程序解释 在C#中创建一个控制台应用程序来管理学生成绩 在这篇文章中,我将向你展示如何使用C#创建…...
鸿蒙开发之ArkUI 界面篇 三十四 容器组件Tabs 自定义TabBar
如果需要修改Tabs的图标和文字之间的距离我们该怎么办呢?好在tabBar是联合类型,提供了自定义tabBar,这里就可以显示特殊图标或者是文字图片,如下图: 这里定义了myBuilder的函数,用了 来修饰,没有…...
AI核身-金融场景凭证篡改检测YOLO原理
引言 YOLO (You Only Look Once) 模型是一种先进的实时目标检测算法,它在计算机视觉领域具有重要的地位。YOLO以其速度和准确性而闻名,能够快速识别图像和视频中的各种物体。这使得它在自动驾驶、安全监控、机器人技术、医学影像分析等众多领域都有着广…...
鹅厂JS面试题——0.1+0.2=0.3吗?
首先公布答案:在JavaScript 中,0.1 0.2 ≠ 0.3 为什么? JavaScript 中的数字使用 IEEE 754 标准的双精度浮点数(64 位)进行表示。这种表示方式在处理十进制小数时,不能精确地表示某些数字。比如0.1 和 0.2 这样的十进…...
软件功能测试重点和流程有哪些?专业软件测评服务公司推荐
软件功能测试就是对产品的各功能进行验证,根据功能测试用例,逐项测试,检查产品是否达到用户要求的功能。功能测试也叫黑盒测试或数据驱动测试,只需考虑需要测试的各个功能,不需要考虑整个软件的内部结构及代码.一般从软…...
【数据结构】AVL树(C++实现)
文章目录 前言AVL树节点的定义AVL树的插入AVL树的旋转AVL树的验证AVL树的删除AVL树的性能与源码 前言 二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,效率低下。因此&…...
AMD新推EPYC与MI325X,挑战英伟达AI市场地位
在人工智能(AI)加速器领域,AMD近日于美国旧金山举办的“推进人工智能”(Advancing AI Event)活动中,宣布了一系列新产品的发布,直接对标英伟达,意图在AI芯片市场占据更大份额。 AMD新…...
wordpress后台更新后 前端没变化的解决方法
使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…...
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以?
Golang 面试经典题:map 的 key 可以是什么类型?哪些不可以? 在 Golang 的面试中,map 类型的使用是一个常见的考点,其中对 key 类型的合法性 是一道常被提及的基础却很容易被忽视的问题。本文将带你深入理解 Golang 中…...
mongodb源码分析session执行handleRequest命令find过程
mongo/transport/service_state_machine.cpp已经分析startSession创建ASIOSession过程,并且验证connection是否超过限制ASIOSession和connection是循环接受客户端命令,把数据流转换成Message,状态转变流程是:State::Created 》 St…...
1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...
【Qt】控件 QWidget
控件 QWidget 一. 控件概述二. QWidget 的核心属性可用状态:enabled几何:geometrywindows frame 窗口框架的影响 窗口标题:windowTitle窗口图标:windowIconqrc 机制 窗口不透明度:windowOpacity光标:cursor…...
CentOS 7.9安装Nginx1.24.0时报 checking for LuaJIT 2.x ... not found
Nginx1.24编译时,报LuaJIT2.x错误, configuring additional modules adding module in /www/server/nginx/src/ngx_devel_kit ngx_devel_kit was configured adding module in /www/server/nginx/src/lua_nginx_module checking for LuaJIT 2.x ... not…...
智能问数Text2SQL Vanna windows场景验证
架构 Vanna 是一个开源 Python RAG(检索增强生成)框架,用于 SQL 生成和相关功能。 机制 Vanna 的工作过程分为两个简单步骤 - 在您的数据上训练 RAG“模型”,然后提出问题,这些问题将返回 SQL 查询,这些查…...
十、【ESP32开发全栈指南: TCP客户端】
一、TCP协议核心特性回顾 TCP与UDP关键差异 特性TCPUDP连接方式面向连接 (三次握手)无连接可靠性可靠传输 (重传/排序/校验)尽力交付数据顺序保证数据按序到达不保证顺序流控制滑动窗口机制无流控制传输效率协议开销大头部开销小适用场景文件传输、网页浏览实时音视频、广播通…...
0x-2-Oracle Linux 9上安装JDK配置环境变量
一、JDK选择和使用 安装完Oracle Linux9.6,同时使用rpm包安装Oracle 23 ai free后, 将面临sqlcl程序无法使用和java无法使用,需要相应进行变量配置问题。 1、java 环境运行不存在,Oracle 23ai free安装后默认安装JDK 11 /opt/…...
API标准的本质与演进:从 REST 架构到 AI 服务集成
在当今数字化浪潮中,API 已成为系统之间沟通与协作的“语言”,REST(Representational State Transfer,表述性状态转移)是一种基于 HTTP 协议的 Web 架构风格。它不仅改变了 Web 应用开发的方式,也成为构建现…...
