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

相关文章

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

SpringBoot 多环境开发实战(从配置、管理与控制)

《SpringBoot多环境开发实战(从配置、管理与控制)》本文详解SpringBoot多环境配置,涵盖单文件YAML、多文件模式、MavenProfile分组及激活策略,通过优先级控制灵活切换环境... 目录一、多环境开发基础(单文件 YAML 版)(一)配置原理与优势(二)实操示例二、多环境开发多文件版