FormUrlEncoded POST 请求,我需要使用 SpringBoot 和 Jackson 将 snake case 值转换为 camelCase

FormUrlEncoded POST request, I need to convert snake case values into camelCase with SpringBoot and Jackson

我正在与第三方供应商集成 API。

我有一个 SpringBoot 和 Jackson 设置

他们向我发送了一个 POST 类型为 formUrlEncoded 且参数在 snake_case 中的请求 (总共超过 10 个参数,没有正文)

例如


POST www.example.com?player_id=somePlayerId&product_id=someProductId&total_amount=totalAmount...

JSON 有很多开箱即用的助手,但我找不到任何 formUrlEncoded 的助手(我希望我遗漏了一些明显的东西)。

我尝试了 @ModelAttribute@RequestParam 但没有成功。

我试图避免 @RequestParam MultiValueMap<String, String> params + 自定义映射器选项

我不认为你遗漏了什么。查看 RequestParamMethodArgumentResolver#resolveName source 我看不到自定义请求参数匹配方式的方法。所以看起来要么你必须实现自己的解析器,要么只用 @RequestParam 注释每个参数并提供名称,例如@RequestParam("product_id") String productId

编辑:

至于ModelAttributeModelAttributeMethodProcessor uses WebDataBinder。同样,您可以使用自定义 DataBinder 对其进行自定义,但我没有发现像 Jackson 那样开箱即用地支持别名。

@RequestParam 是最简单的方法,它允许您定义查询参数的确切名称,例如:

@PostMapping
public String foo(@RequestParam("player_id") String playerId){
    

}

如果要将所有查询参数绑定到一个对象,则必须使用@ModelAttribute。它基于 DataBinder,与杰克逊无关。默认情况下,它只支持将查询参数绑定到字段与查询参数同名的对象。所以你可以考虑将查询参数绑定到以下对象:

public class Request {
    private String player_id;
    private String product_id;
    private Long total_amount;
}

如果您真的想从具有蛇形大小写值的查询参数绑定到遵循传统 java 命名约定(即小驼峰大小写)的对象,则必须自定义 WebDataBinder

想法是覆盖其 addBindValues() 并检查查询参数名称是否为蛇形格式,将其转换为小驼峰格式并将其添加为请求的绑定值。像 :

public class MyServletRequestDataBinder extends ExtendedServletRequestDataBinder {

    private static Converter<String, String> snakeCaseToLowerCamelConverter = CaseFormat.LOWER_UNDERSCORE
            .converterTo(CaseFormat.LOWER_CAMEL);

    public MyServletRequestDataBinder(Object target) {
        super(target);
    }

    public MyServletRequestDataBinder(Object target, String objectName) {
        super(target, objectName);
    }

    @Override
    protected void addBindValues(MutablePropertyValues mpvs, ServletRequest request) {
        super.addBindValues(mpvs, request);

        Enumeration<String> paramNames = request.getParameterNames();
        while (paramNames != null && paramNames.hasMoreElements()) {
            String paramName = paramNames.nextElement();
            
            if(paramName.contains("_")) {
                String[] values = request.getParameterValues(paramName);
                if (values == null || values.length == 0) {
                    // Do nothing, no values found at all.
                } else if (values.length > 1) {
                    mpvs.addPropertyValue(snakeCaseToLowerCamelConverter.convert(paramName), values);
                } else {
                    mpvs.addPropertyValue(snakeCaseToLowerCamelConverter.convert(paramName), values[0]);
                }
            }
        }
    }
}

P.S 我正在使用 Guava 帮助我将蛇形大小写转换为 lowerCamelCase。

但是为了使用自定义的 WebDataBinder ,你必须依次自定义 WebDataBinderFactoryRequestMappingHandlerAdapter 因为:

  • 自定义 WebDataBinderFactory 以创建自定义的 WebDataBinder
  • 自定义 RequestMappingHandlerAdapter 以创建 WebDataBinderFactory

类似于:

public class MyServletRequestDataBinderFactory extends ServletRequestDataBinderFactory {

        public MyServletRequestDataBinderFactory(List<InvocableHandlerMethod> binderMethods,
                WebBindingInitializer initializer) {
            super(binderMethods, initializer);
        }

        @Override
        protected ServletRequestDataBinder createBinderInstance(Object target, String objectName,
                NativeWebRequest request) throws Exception {
            return new MyServletRequestDataBinder(target, objectName);
        }

    }

public class MyRequestMappingHandlerAdapter extends RequestMappingHandlerAdapter {

    @Override
    protected InitBinderDataBinderFactory createDataBinderFactory(List<InvocableHandlerMethod> binderMethods)
            throws Exception {
        return new MyServletRequestDataBinderFactory(binderMethods, getWebBindingInitializer());
    }

}
    

最后在您的配置中注册使用自定义 RequestMappingHandlerAdapter :

@Configuration
public class Config extends DelegatingWebMvcConfiguration {

    @Override
    protected RequestMappingHandlerAdapter createRequestMappingHandlerAdapter() {
        return new MyRequestMappingHandlerAdapter();
    }
}