SpringBoot之HiddenHttpMethodFilter

2024-06-12 17:28

本文主要是介绍SpringBoot之HiddenHttpMethodFilter,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

默认情况下,表单只能发送GET、POST请求,如果需要发送PUT、DELETE请求呢?SpringBoot通过 HiddenHttpMethodFilter 解决了这一问题。

演示表单只能发送GET、POST请求

创建user.html
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><form action="/user" method="get"><input value="GET 提交" type="submit"/></form><form action="/user" method="post"><input value="POST提交" type="submit"/></form><form action="/user" method="put"><input value="DELETE 提交" type="submit"/></form><form action="/user" method="delete"><input value="PUT 提交" type="submit"/></form>
</body>
</html>
页面展示

创建 UserController
@RestController
public class UserController {@GetMapping("/user")public String getUser() {return "get user";}@PostMapping("/user")public String postUser() {return "post user";}@PutMapping("/user")public String putUser() {return "put user";}@DeleteMapping("/user")public String deleteUser() {return "delete user";}
}
发送 GET 请求

发送 POST 请求 

发送 DELETE 请求 

发送 PUT 请求

我们期望的DELETE、PUT 请求,都转发到GET请求了

通过 HiddenHttpMethodFilter 发送DELETE、PUT 请求

添加配置
spring:mvc:hiddenmethod:filter:enabled: true
修改 user.html
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><form action="/user" method="get"><input value="GET 提交" type="submit"/></form><form action="/user" method="post"><input value="POST 提交" type="submit"/></form><form action="/user" method="post"><input name="_method" type="hidden" value="DELETE"/><input value="DELETE 提交" type="submit"/></form><form action="/user" method="post"><input name="_method" type="hidden" value="PUT"/><input value="PUT 提交" type="submit"/></form>
</body>
</html>
发送 DELETE 请求 

发送 PUT 请求

 可以正确发送DELETE、PUT 请求

源码简析

WebMvcAutoConfiguration#hiddenHttpMethodFilter
@Bean
@ConditionalOnMissingBean(HiddenHttpMethodFilter.class)
@ConditionalOnProperty(prefix = "spring.mvc.hiddenmethod.filter", name = "enabled")
public OrderedHiddenHttpMethodFilter hiddenHttpMethodFilter() {return new OrderedHiddenHttpMethodFilter();
}

通过 @ConditionalOnProperty 注解,我们明白需要将 spring.mvc.hiddenmethod.filter.enable 属性设置为 true。

OrderedHiddenHttpMethodFilter 类结构

通过 OrderedHiddenHttpMethodFilter 的类结构,我们可以看出来它继承 HiddenHttpMethodFilter,我们查看相关源码。

public class HiddenHttpMethodFilter extends OncePerRequestFilter {private static final List<String> ALLOWED_METHODS =Collections.unmodifiableList(Arrays.asList(HttpMethod.PUT.name(),HttpMethod.DELETE.name(), HttpMethod.PATCH.name()));/** Default method parameter: {@code _method}. */public static final String DEFAULT_METHOD_PARAM = "_method";private String methodParam = DEFAULT_METHOD_PARAM;/*** Set the parameter name to look for HTTP methods.* @see #DEFAULT_METHOD_PARAM*/public void setMethodParam(String methodParam) {Assert.hasText(methodParam, "'methodParam' must not be empty");this.methodParam = methodParam;}@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)throws ServletException, IOException {HttpServletRequest requestToUse = request;if ("POST".equals(request.getMethod()) && request.getAttribute(WebUtils.ERROR_EXCEPTION_ATTRIBUTE) == null) {String paramValue = request.getParameter(this.methodParam);if (StringUtils.hasLength(paramValue)) {String method = paramValue.toUpperCase(Locale.ENGLISH);if (ALLOWED_METHODS.contains(method)) {requestToUse = new HttpMethodRequestWrapper(request, method);}}}filterChain.doFilter(requestToUse, response);}/*** Simple {@link HttpServletRequest} wrapper that returns the supplied method for* {@link HttpServletRequest#getMethod()}.*/private static class HttpMethodRequestWrapper extends HttpServletRequestWrapper {private final String method;public HttpMethodRequestWrapper(HttpServletRequest request, String method) {super(request);this.method = method;}@Overridepublic String getMethod() {return this.method;}}}
注意点
  1. DEFAULT_METHOD_PARAM : _method
  2. ALLOWED_METHODS : PUTDELETEPATCH
  3. HttpMethodRequestWrappergetMethod 方法

通过上述源码我们得出以下结论:如果表单是通过 POST 提交的且存在参数 methodParam (默认 _method),methodParam 参数值如果是 ALLOWED_METHODS 中的 一种,则将HttpServletRequest 封装成 HttpMethodRequestWrapper ,这个类会重写 getMethod 方法,方法的返回值就是 methodParam 参数传递过来的值。

请求映射
DispatcherServlet#doDispatch

DispatcherServlet#getHandler

请求一般是由 RequestMappingHandlerMapping 处理的 

 AbstractHandlerMapping#getHandler

RequestMappingInfoHandlerMapping#getHandlerInternal

AbstractHandlerMethodMapping#getHandlerInternal

AbstractHandlerMethodMapping#lookupHandlerMethod

URI 和具体方法的映射关系,都存储在 mappingRegistry 这个属性中

请求映射的详细解析可以查看这篇博文 : SpringBoot之请求映射原理

扩展:自定义HiddenHttpMethodFilter

如果存在自定义的 HiddenHttpMethodFilter,则默认的 HiddenHttpMethodFilter 失效。

创建类 CustomHiddenHttpMethodFilter 
public class CustomHiddenHttpMethodFilter extends HiddenHttpMethodFilter {
}
创建配置类 FilterConfig
@Configuration
public class FilterConfig {@Beanpublic CustomHiddenHttpMethodFilter hiddenHttpMethodFilter() {CustomHiddenHttpMethodFilter customHiddenHttpMethodFilter = new CustomHiddenHttpMethodFilter();customHiddenHttpMethodFilter.setMethodParam("_custom");return customHiddenHttpMethodFilter;}
}
修改user.html
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body><form action="/user" method="get"><input value="GET 提交" type="submit"/></form><form action="/user" method="post"><input value="POST 提交" type="submit"/></form><form action="/user" method="post"><input name="_custom" type="hidden" value="DELETE"/><input value="DELETE 提交" type="submit"/></form><form action="/user" method="post"><input name="_custom" type="hidden" value="PUT"/><input value="PUT 提交" type="submit"/></form></body>
</html>
发送 DELETE 请求 

发送 PUT 请求

自定义参数生效, 可以正确发送DELETE、PUT 请求

这篇关于SpringBoot之HiddenHttpMethodFilter的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1054856

相关文章

Spring Boot中的路径变量示例详解

《SpringBoot中的路径变量示例详解》SpringBoot中PathVariable通过@PathVariable注解实现URL参数与方法参数绑定,支持多参数接收、类型转换、可选参数、默认值及... 目录一. 基本用法与参数映射1.路径定义2.参数绑定&nhttp://www.chinasem.cnbs

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建

Java中Integer128陷阱

《Java中Integer128陷阱》本文主要介绍了Java中Integer与int的区别及装箱拆箱机制,重点指出-128至127范围内的Integer值会复用缓存对象,导致==比较结果为true,下... 目录一、Integer和int的联系1.1 Integer和int的区别1.2 Integer和in

SpringSecurity整合redission序列化问题小结(最新整理)

《SpringSecurity整合redission序列化问题小结(最新整理)》文章详解SpringSecurity整合Redisson时的序列化问题,指出需排除官方Jackson依赖,通过自定义反序... 目录1. 前言2. Redission配置2.1 RedissonProperties2.2 Red

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.

JSONArray在Java中的应用操作实例

《JSONArray在Java中的应用操作实例》JSONArray是org.json库用于处理JSON数组的类,可将Java对象(Map/List)转换为JSON格式,提供增删改查等操作,适用于前后端... 目录1. jsONArray定义与功能1.1 JSONArray概念阐释1.1.1 什么是JSONA

Java JDK1.8 安装和环境配置教程详解

《JavaJDK1.8安装和环境配置教程详解》文章简要介绍了JDK1.8的安装流程,包括官网下载对应系统版本、安装时选择非系统盘路径、配置JAVA_HOME、CLASSPATH和Path环境变量,... 目录1.下载JDK2.安装JDK3.配置环境变量4.检验JDK官网下载地址:Java Downloads

Spring boot整合dubbo+zookeeper的详细过程

《Springboot整合dubbo+zookeeper的详细过程》本文讲解SpringBoot整合Dubbo与Zookeeper实现API、Provider、Consumer模式,包含依赖配置、... 目录Spring boot整合dubbo+zookeeper1.创建父工程2.父工程引入依赖3.创建ap