[TOC]
# 1. 简单获取
HttpServletRequest 提供了下面 4 个方法来获取简单的请求体。
```java
//根据指定的参数名获取对应的值
//1. 如果不存在则返回 null
//2. 如果存在但不设置值返回空字符串 ""
//3. 如果存在相同的参数名则返回第一个的值。
String getParameter(String name)
//返回相同参数名对应的值
String[] getParameterValues(String name)
//返回所有的参数名
Enumeration<String> getParameterNames()
//返回参数并包装为一个Map,key就是参数名
Map<String, String[]> getParameterMap()
```
上面的 4 个基本方法有如下限制:
```
1. 能获取拼接在url后面的参数,如username=zhangsan&password=123456789
2. 能获取请求类型为 multipart/form-data 的请求体,但是不能获取上传的文件
3. 能获取请求类型为 application/x-www-form-urlencoded 的请求体
4. 不能获取请求类型为 application/json 的请求体
```
# 2. 允许HttpServletRequest流读取多次
HttpServletRequest 的流默认只能读取一次,如果第二次读取就会抛出异常`has already been called for this reques`。
```java
HttpServletRequest request = ...
//第一次读取BufferedReader、或ServletInputStream
BufferedReader reader = request.getReader();
ServletInputStream in = request.getInputStream();
//如果再读取第二次,就不行了
//BufferedReader reader = request.getReader();
//ServletInputStream in = request.getInputStream();
```
如果需要可以重复多次读取,可以如下封装:
```java
public class MultiReadHttpServletRequest extends HttpServletRequestWrapper {
private final byte[] body;
public MultiReadHttpServletRequest(HttpServletRequest request) throws IOException {
super(request);
try (InputStream inputStream = request.getInputStream();
ByteArrayOutputStream buffer = new ByteArrayOutputStream()) {
byte[] bytes = new byte[1024];
int len;
while ((len = inputStream.read(bytes)) != -1) {
buffer.write(bytes, 0, len);
}
body = buffer.toByteArray();
}
}
@Override
public ServletInputStream getInputStream() throws IOException {
final ByteArrayInputStream bais = new ByteArrayInputStream(body);
return new ServletInputStream() {
@Override
public boolean isFinished() {
return false;
}
@Override
public boolean isReady() {
return false;
}
@Override
public void setReadListener(ReadListener readListener) {
}
@Override
public int read() throws IOException {
return bais.read();
}
};
}
@Override
public BufferedReader getReader() throws IOException {
return new BufferedReader(new InputStreamReader(this.getInputStream()));
}
}
```
在拦截器拦截之前、或过滤器过滤之前如下覆盖默认的HttpServletRequest:
```java
HttpServletRequest request = new MultiReadHttpServletRequest(request);
```
# 3. 获取application/json请求体
当请求类型为`application/json` ,`getParameterXXX`方法就不能获取请求体了,如果想要获取,可以如下获取。
```java
public class TestFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
boolean isJSON = ("" + request.getContentType()).toLowerCase().contains(MediaType.APPLICATION_JSON_VALUE);
if (isJSON) {
//使用上面封装的MultiReadHttpServletRequest
request = new MultiReadHttpServletRequest(request);
//在过滤前获取
testGetJson(request);
}
filterChain.doFilter(request, servletResponse);
//也可以同时在响应后获取
if (isJSON) {
testGetJson(request);
}
}
private void testGetJson(HttpServletRequest request) throws IOException {
BufferedReader reader = request.getReader();
StringBuilder jsonData = new StringBuilder();
String line;
while ((line = reader.readLine()) != null) {
jsonData.append(line);
}
//去除所有的空字符串
String formatStr = jsonData.toString().replaceAll("\\s+", "");
ObjectMapper objectMapper = new ObjectMapper();
Map<String, Object> jsonMap = objectMapper.readValue(formatStr, Map.class);
log.info(objectMapper.writeValueAsString(jsonMap));
}
}
```
# 4. 获取multipart/form-data请求体
当请求类型为 multipart/form-data ,`getParameterXXX`方法不能获取上传的文件信息。
可以采用下面的方案来获取完整的 multipart/form-data 请求体。
```java
HttpServletRequest request = (HttpServletRequest) servletRequest;
StandardServletMultipartResolver resolver = new StandardServletMultipartResolver();
if (resolver.isMultipart(request)) {
MultipartHttpServletRequest multReq = resolver.resolveMultipart(request);
//获取文本类型数据
for (Enumeration<String> names = multReq.getParameterNames(); names.hasMoreElements(); ) {
String name = names.nextElement();
log.info("[{}={}", name, request.getParameter(name));
}
//获取文件类型数据
List<MultipartFile> files = multReq.getFiles("file");
for (MultipartFile file : files) {
log.info(file.getOriginalFilename());
//如果需要在多个地方存储文件,则不能调用transferTo方法,否则文件只能被存储一次就没了
//应该使用file.getInputStream()取出文件
//file.transferTo(...);
try (InputStream inputStream = file.getInputStream()) {
File destinationFile = new File("F:\\demo\\f_" + file.getOriginalFilename());
Files.copy(inputStream, destinationFile.toPath());
} catch (IOException e) {
e.printStackTrace();
}
}
}
```
- Spring
- Spring是什么
- Spring与EJB对比
- Spring的组成
- 首个Spring程序
- IoC控制反转
- 什么是IoC
- IoC编程
- 依赖注入方式
- 不同变量注入
- AOP面向切面编程
- AOP思想
- AOP实现原理
- AOP关键术语
- AOP编程
- 5种增强方式
- 切入点规则
- 自动装配
- Spring注解开发
- Bean注解
- AOP注解
- 完全注解
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean作用域
- Bean生命周期
- SpringBoot
- SpringBoot是什么
- 项目创建
- 配置文件
- 配置类型
- 读取配置
- 占位符
- 多环境配置
- 配置优先级
- 更改配置文件
- 自定义IoC容器
- 常用组件
- ApplicationContextAware
- CommandLineRunner
- Boot[Web]
- 引入模板引擎
- 静态资源访问
- 指定首页
- JSP支持
- 注册拦截器
- 注册Servlet组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 拦截器与过滤器区别
- 文件上传
- 文件下载
- 变更服务器
- Controller层封装
- HttpServletRequest
- 获取请求行
- 获取请求头
- 获取请求体
- Boot[自动配置]
- 自动配置是什么
- 自动配置报告
- 关闭自动配置
- 条件注解
- Boot[场景启动器]
- 场景启动器是什么
- 自定义场景启动器
- Boot[日志]
- 日志框架
- 日志级别
- 日志配置
- 配置文件
- 切换日志
- Boot[邮件任务]
- Boot[定时任务]
- cron表达式
- 起步
- 任务并行
- 注解Scheduled参数
- Boot[异步任务]
- 起步
- 注意事项与原理
- 自定义线程池
- Boot[缓存]
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 起步
- 自定义key生成器
- 工作原理
- Boot[Redis]
- 起步
- 序列化机制
- Boot[Jdbc]
- 起步
- 两个模板类
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- 自定义JdbcTemplate
- Boot[JPA]
- SpringDataJPA是什么
- 与JPA、Hibernate的关系
- 起步
- SpringDataJPA原理
- 查询方式
- 方法命名规则查询
- 限制查询结果查询
- 注解Query查询
- 命名参数查询
- SpEL表达式查询
- 原生查询
- 更新与删除
- 查询指定字段
- Specification动态查询
- 分页查询与排序
- 多表查询
- 一对一查询
- 一对多查询
- 多对多查询
- Specification查询
- Query注解查询
- 主键策略
- 单独主键
- 联合主键
- 级联操作
- 加载规则
- 审计功能
- 常用注解
- 避坑指南
- Boot[JSR303]
- JSR303是什么
- 常用约束
- 起步
- 简单校验
- 嵌套校验
- 分组校验
- 自定义约束注解
- 自定义校验工具
- Spring事务
- 事务的作用
- 起步
- 事务参数
- SpringDoc文档
- SpringDoc是什么
- 起步
- 自定义配置
- 常用Doc注解
- JSR303文档
- knife4j文档
- 常用配置
- Boot[RabbitMQ]
- 起步
- Fanout交换机类型
- Direct交换机类型
- Topic交换机类型
- 延迟队列插件
- RabbitListener监听方法
- JWT认证
- 认证流程
- 起步
- 密码加密
- JWT认证实现