当我们在测试环境中进行压力测试时,我遇到了一个奇怪的情况。当应用程序负载较高时,查询字符串偶尔会丢失,Spring 会抛出
MissingServletRequestParameterException
。
为了找到根本原因,我在最前面添加了一些日志Filter
(代码如下),然后奇怪的事情发生了。
public static void printRequestParameter(HttpServletRequest request) {
Map<String, String[]> parameterMap = request.getParameterMap();
log.info("Request URI : {}, method = {} , query string = {}", request.getRequestURI(), request.getMethod(), request.getQueryString());
if (MapUtils.isNotEmpty(parameterMap)) {
parameterMap.forEach((k, v) -> {
log.info("Request parameter name = {}, value = {}", k, ArrayUtils.isEmpty(v) ? Strings.EMPTY : Arrays.stream(v).collect(Collectors.joining(COMMA)));
});
}
}
request.getParameterMap()
为空,但是,查询字符串不为空,我得到了如下日志:
Request URI : /a/b/c, method = POST , query string = foo=bar.
但没有类似的日志:
Request parameter name = foo , value = bar
而我们的
Controller
使用@RequestParam() String foo
来接收参数,最后,Spring抛出
MissingServletRequestParameterException Handler org.springframework.web.bind.MissingServletRequestParameterException: Required String parameter 'foo' is not present
at org.springframework.web.method.annotation.RequestParamMethodArgumentResolver.handleMissingValue(RequestParamMethodArgumentResolver.java:204)
at org.springframework.web.method.annotation.AbstractNamedValueMethodArgumentResolver.resolveArgument(AbstractNamedValueMethodArgumentResolver.java:114)
at org.springframework.web.method.support.HandlerMethodArgumentResolverComposite.resolveArgument(HandlerMethodArgumentResolverComposite.java:121)
at org.springframework.web.method.support.InvocableHandlerMethod.getMethodArgumentValues(InvocableHandlerMethod.java:167)
at org.springframework.web.method.support.InvocableH
我只是想知道,为什么查询字符串中的参数不包含在
parameterMap
中?
笔记:
奇怪的行为只是偶尔发生,大多数时候它只是有效的。
我的spring boot版本是2.3.9.RELEASE
,嵌入式tomcat版本是9.0.43
。
如有任何帮助,我们将不胜感激!
ServletRequest.getParameterMap
抛出任何异常,因此任何参数解析失败都会导致参数列表为空。
"org.apache.catalina.parameter_parse_failed_reason"
属性。
应该失败的查询字符串示例:
?=noname
,?urlEncoding=%ue
最后,这是
@Async
问题,有人将此注释放在了Controller类上