最近很多交互要同原生的HttpServletRequest
和HttpServletResponse
打交道。从HttpServletRequest
中读取body数据封装成某种数据结构;向HttpServletResponse
写入数据并响应。传统的写法非常不优雅,今天给大家介绍一种比较优雅的方式。
HttpMessageConverter
HttpMessageConverter
是Spring框架提供的一个消息转换器模型,用于在 HTTP 请求和响应之间进行转换的策略接口。它可以对输入消息HttpInputMessage
进行读;也可以对输出消息HttpOutputMessage
进行写。
文章图片
Spring MVC的消息转换都是通过这个接口的实现来完成的。HttpMessageConverter
有很多实现:
文章图片
通常Spring MVC中处理Form表单提交、JSON、XML、字符串、甚至Protobuf都由HttpMessageConverter
的实现来完成,前端传递到后端的body参数,后端返回给前端的数据都是由这个接口完成转换的。在Spring IoC中(Spring MVC环境)还存在一个存放HttpMessageConverter
的容器HttpMessageConverters
:
@Bean
@ConditionalOnMissingBean
public HttpMessageConverters messageConverters(ObjectProvider> converters) {
return new HttpMessageConverters((Collection)converters.orderedStream().collect(Collectors.toList()));
}
我们可以直接拿来使用。那么到底怎么使用呢?那首先要搞清楚
HttpInputMessage
和HttpOutputMessage
是干什么用的。HttpInputMessage
HttpInputMessage
表示一个 HTTP 输入消息,由请求头headers和一个可读的请求体body组成,通常由服务器端的 HTTP 请求句柄或客户端的 HTTP 响应句柄实现。文章图片
而
HttpServletRequest
是ServletRequest
的扩展接口,提供了HTTP Servlet的请求信息,也包含了请求头和请求体,所以两者是有联系的。我们只要找出两者之间的实际关系就能让HttpMessageConverter
去读取并处理HttpServletRequest
携带的请求信息。ServletServerHttpRequest 说实话还真找到了:
文章图片
ServletServerHttpRequest
不仅仅是HttpInputMessage
的实现,它还持有了一个HttpServletRequest
实例属性,ServletServerHttpRequest
的所有操作都是基于HttpServletRequest
进行的。我们可以通过构造为其注入HttpServletRequest
实例,这样HttpMessageConverter
就能间接处理HttpServletRequest
了。提取请求体实战
这里聚焦的场景是在Servlet过滤器中使用
HttpMessageConverter
,在Spring MVC中不太建议去操作HttpServletRequest
。我选择了FormHttpMessageConverter
,它通常用来处理application/x-www-form-urlencoded
请求。我们编写一个过滤器来拦截请求提取body:/**
* 处理 application/x-www-form-urlencoded 请求
*
* @authorfelord.cn
*/@Component
public class FormUrlencodedFilter implements Filter {
private final FormHttpMessageConverter formHttpMessageConverter = new FormHttpMessageConverter();
private static final Logger log = LoggerFactory.getLogger(FormUrlencodedFilter.class);
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException {
String contentType = request.getContentType();
MediaType type= StringUtils.hasText(contentType)? MediaType.valueOf(contentType):null;
ServletServerHttpRequest serverHttpRequest = new ServletServerHttpRequest((HttpServletRequest) request);
if (formHttpMessageConverter.canRead(MultiValueMap.class,type)) {
MultiValueMap read = formHttpMessageConverter.read(null, serverHttpRequest);
log.info("打印读取到的请求体:{}",read);
}
}
}
然后执行一个
POST
类型,Content-Type
为application/x-www-form-urlencoded
的请求:POST /ind HTTP/1.1
Host: localhost:8080
Content-Type: application/x-www-form-urlencoded
Content-Length: 20a=b123&c=d123&e=f123
控制台会打印:
2021-12-30 6:43:56.409INFO 12408 --- [nio-8080-exec-1] sfds: 打印读取到的请求体:{a=[b123], c=[d123], e=[f123]}
ServletServerHttpResponse 有
ServletServerHttpRequest
就有ServletServerHttpResponse
,大致原理差不多。它正好和ServletServerHttpRequest
相反,如果我们需要去处理响应问题,比如想通过HttpServletResponse
写个JSON响应,大概可以这么写:ServletServerHttpResponse servletServerHttpResponse = new ServletServerHttpResponse(response);
// 使用json converter
MappingJackson2HttpMessageConverter mappingJackson2HttpMessageConverter = new MappingJackson2HttpMessageConverter();
//authentication 指的是需要写的对象实例
mappingJackson2HttpMessageConverter.write(authentication, MediaType.APPLICATION_JSON,servletServerHttpResponse);
总结
HttpMessageConverter
抽象了HTTP消息转换的策略,可以帮助我们优雅地处理一些请求响应的问题。不过有一点需要注意,请求体body只能读取一次,即使它包裹在ServletServerHttpRequest
中,要注意和HttpServletRequestWrapper
的区别。关注公众号:Felordcn 获取更多资讯
【如何优雅地读写HttpServletRequest和HttpServletResponse的请求体】个人博客:https://felord.cn
推荐阅读
- Java|Java基础——数组
- 人工智能|干货!人体姿态估计与运动预测
- java简介|Java是什么(Java能用来干什么?)
- Java|规范的打印日志
- Linux|109 个实用 shell 脚本
- 程序员|【高级Java架构师系统学习】毕业一年萌新的Java大厂面经,最新整理
- Spring注解驱动第十讲--@Autowired使用
- SqlServer|sql server的UPDLOCK、HOLDLOCK试验
- jvm|【JVM】JVM08(java内存模型解析[JMM])
- 技术|为参加2021年蓝桥杯Java软件开发大学B组细心整理常见基础知识、搜索和常用算法解析例题(持续更新...)