问题描述

小伙伴在开发某个http接口时,发现controller层接收不到post表单提交的数据,直接在请求地址后面添加params参数可以接收到,post的json参数也可以接收到

@PostMapping("/test")
    public String test(String token, HttpServletRequest request) {
        logger.info("token:" + token);
        return "token:" + token;
    }

postman发送请求如下

controller怎么接收数组java_spring


发现token获取不到,一直为null

原因分析及排查:

猜想问题可能产生的原因:

  • 某些拦截器特殊处理了请求
    排查所有拦截器未发现异常,甚至放行该请求也不行
  • 项目某些配置设置不正确或未开启
    检查spring的配置文件,application.yml或者application.properties等配置文件,查看servlet相关配置是否开启。查看到请求的content-type是multipart/form-data,便查看是否有如下配置,默认为true,开启表单文件上传,查看发现没有特别设置
spring.servlet.multipart.enabled=true
  • springboot版本原因,项目使用1.5.x版本
    检查请求源码,debug断点调试,从servlet分发到controller各个链路,未找到具体原因
  • 其他

解决方案:

其实上面检查某些配置的步骤就很接近答案了,最后是这样确定问题产生原因的:将项目的所在模块copy一份,去掉所有拦截器等配置等,只留着启动类和controller,pom中的依赖只留下springboot核心的依赖,删除一切特殊配置如启动类下的某些bean,发现controller能接收到表单提交数据,重新加上启动类的bean后,发现问题复现,最终确定问题为启动类下添加的bean有问题,如下:

@Bean
    public ServletRegistrationBean<CXFServlet> dispatcherServlet() {
        ServletRegistrationBean<CXFServlet> servletRegistrationBean = new ServletRegistrationBean<>(new CXFServlet(), "/services/*");
        servletRegistrationBean.setLoadOnStartup(1);
        return servletRegistrationBean;
    }

    @Bean
    public ServletRegistrationBean restServlet(){
        //注解扫描上下文
        AnnotationConfigWebApplicationContext applicationContext
                = new AnnotationConfigWebApplicationContext();
        //base package
        applicationContext.scan("com.*");
        //通过构造函数指定dispatcherServlet的上下文
        DispatcherServlet rest_dispatcherServlet
                = new DispatcherServlet(applicationContext);

        //用ServletRegistrationBean包装servlet
        ServletRegistrationBean registrationBean
                = new ServletRegistrationBean(rest_dispatcherServlet);
        registrationBean.setLoadOnStartup(1);
        //指定urlmapping
        registrationBean.addUrlMappings("/*");
        //指定name,如果不指定默认为dispatcherServlet
        registrationBean.setName("rest");
        return registrationBean;
    }

此处第一个bean注册了一个servlet,名称为dispatcherServlet,从代码可以看出,这个bean目的是给webservice服务用来分发请求的。坏就坏在这里,这个dispatcherServlet与springboot本身的dispatcherServlet重名了!!!springboot本身的dispatcherServlet是用来分发http请求到controller层的。

我们可以合理猜测,原开发者是想将项目用作了webservice服务端,但这样一来却覆盖了springboot中dispatcherServlet,发现controller层请求异常后,下面又注册了一个bean试图解决controller的请求异常问题,但这忽略了我们遇到的问题:虽然controller层接口能请求到,却接收不到表单提交的数据了。

知其然,还要知其所以然。下面还要分析下两者的差异,为什么自己创建的dispatcherServlet会获取不到表单的属性???

查看调用链

controller怎么接收数组java_后端_02


追踪源码,发现tomcat中的coyoteRequest中的parameters属性两者是有差异的,一个可以获取到表单中的token,一个没有

controller怎么接收数组java_后端_03


controller怎么接收数组java_spring_04


那么继续,追踪tomcat源码,最终定位到Request源码,在ApplicationFilterChain进行多个过滤器过滤处理时,在执行到HiddenHttpMethodFilter时,

@Override
	protected 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)) {
				requestToUse = new HttpMethodRequestWrapper(request, paramValue);
			}
		}

		filterChain.doFilter(requestToUse, response);
	}

其中getParameter方法,会调用ApplicationHttpRequest中的getParameter方法

@Override
    public String getParameter(String name) {

        parseParameters();

        String[] value = parameters.get(name);
        if (value == null) {
            return null;
        }
        return value[0];

    }

其中的parseParameters()方法中的逻辑,则是导致差异的根本原因在处理表单提交时,会进行parseParts操作

protected void parseParameters() {
        parametersParsed = true;
        ....
  		if ("multipart/form-data".equals(contentType)) {
            parseParts(false);
            success = true;
            return;
        }
        ....

parseParts方法如下:

controller怎么接收数组java_文件大小_05


controller怎么接收数组java_表单_06


发现差异在MultipartConfigElement这个类上,使用spring默认的dispatcherServlet是有值的,而自己创建的是null,所以问题的根源就在这,再经过一番查询,得知这个类的作用:

在Tomcat的Request源码中,getWrapper().getMultipartConfigElement() 这段代码是用于获取当前请求对应的Servlet包装器(通常是StandardWrapper)的MultipartConfigElement。MultipartConfigElement是Java Servlet 3.0规范中引入的一个接口,它定义了如何处理multipart/form-data类型的请求(通常是文件上传)。

具体来说,MultipartConfigElement提供了以下配置信息:

getLocation():指定存储上传文件的临时位置。
getMaxFileSize():指定单个上传文件的最大大小(以字节为单位)。
getMaxRequestSize():指定整个请求的最大大小(包括所有上传的文件)(以字节为单位)。
getFileSizeThreshold():指定在内存中存储文件数据的大小阈值。如果上传的文件大小小于这个阈值,那么文件数据会被存储在内存中,否则会被存储在磁盘上。
当Tomcat处理一个multipart/form-data类型的请求时(如文件上传),它会检查MultipartConfigElement中的配置信息来决定如何处理这个请求。例如,如果上传的文件大小超过了getMaxFileSize()或getMaxRequestSize()中指定的值,Tomcat会抛出一个异常来拒绝这个请求。

故此我们需要在创建servlet时,就需要配置MultipartConfigElement这个类

@Bean
    public ServletRegistrationBean restServlet(){
        //注解扫描上下文
        AnnotationConfigWebApplicationContext applicationContext
                = new AnnotationConfigWebApplicationContext();
        //base package
        applicationContext.scan("com.*");
        //通过构造函数指定dispatcherServlet的上下文
        DispatcherServlet rest_dispatcherServlet
                = new DispatcherServlet(applicationContext);

        //用ServletRegistrationBean包装servlet
        ServletRegistrationBean registrationBean
                = new ServletRegistrationBean(rest_dispatcherServlet);
        registrationBean.setLoadOnStartup(1);
        //指定url_mapping
        registrationBean.addUrlMappings("/*");
        //指定name,如果不指定默认为dispatcherServlet
        registrationBean.setName("rest");

        // 配置MultipartConfigElement
        MultipartConfigElement multipartConfigElement = new MultipartConfigElement(
                "D:\\temp", // 文件存储的临时位置
                2097152L, // 单个文件最大大小,单位字节,这里是2MB
                4194304L, // 请求的最大大小,单位字节,这里是4MB
                10240 // 文件大小阈值,超过这个值会存储在磁盘上,单位字节,这里是10KB
        );
        registrationBean.setMultipartConfig(multipartConfigElement);
        return registrationBean;
    }

最后,调用postman发现,成功在controller层获取到了表单中的属性值!!!

总结:

这个问题应该在一个月前就遇到了,当时没有找到根本原因,但心里会一直想着,今天总算找到原因了。
所以,能用spring默认的就用默认的配置,自己配置可能根本不知道会漏了哪些东西,导致一些小问题的产生。其实根本原因就是复制粘贴代码时,需要考虑到对项目产生的影响。