spring|spring mvc 关于对optional的处理和自定义处理

springmvc4.1+能够支持optional参数,但不支持对象内field的optional。
举个例子

//对于这种直接在方法上定义的optional的基本类型或者引用类型 //比如参数里没有id,或者没有obj,那么spring mvc会设置一个空的Optional对象 //咱们可以通过Optional对象进行空判断 @GetMapping("/get") public void test1(Optional id, Optional obj) { id.ifPresent(i->{}); } //但是对于引用类型里field的Optional的话,spring mvc不支持 //定义一个对象,里面属性用Optional类型 @Data public class TestVo { Optional id; Optional name; } //如果这时候参数传了id,那么testVo.getId().get()可以拿到值 //但是如果没有传name,这时的testVo的name属性为null而不是Optional的empty类型 @GetMapping("/get") public Object test(TestVo testVo) { System.err.println(testVo.getId().get()); System.err.println(testVo.getName().isPresent()); //报空指针异常 return testVo; }
咱们来看看springmvc是怎么处理的?
对于方法参数,spring mvc是如何赋值的呢?
mvc有个RequestMappingHandlerAdapter,名称可以看出请求映射处理适配,就是它匹配我们访问的url路径和controller的requestMapping路径的,其中包含了一个HandlerMethodReturnValueHandler,处理结果赋值,以及HandlerMethodArgumentResolver(咱们讲这个),处理方法参数解析,就是由它进行参数赋值的,它有很多实现类,比如PathVariableMapMethodArgumentResolver就是咱们在参数上定义一个 @PathVariable时进行赋值的实现,RequestParamMapMethodArgumentResolver就是咱们在参数上定义一个 @RequestParam时进行赋值的实现。
那么平时默认的没有注解的引用类型,是通过哪个解析器呢?通过断点可知是通过ServletModelAttributeMethodProcessor来实现的。里面有WebDataBinder(题外话,WebDataBinder里有个ConversionService,这是spring提供的一个转换接口,底层使用BeanWrapper进行bean的操作),实际实现类是ExtendedServletRequestDataBinder(该类就是进行数据绑定的),bind方法里就是获取request的参数列表,由于我们只传了id,没有传name,所以这里返回的MutablePropertyValues时里面就只有id(问题就出在这里),并设置id的值。
对于引用类型会使用到PropertyDescriptor(一系列的,属于java bean的规范)类设置列属性的值,TypeDescriptor记录属性类型,然后会根据属性名和参数类型获取PropertyEditor(PropertyEditor editor = this.propertyEditorRegistry.findCustomEditor(requiredType, propertyName)),如果editor为空的话,就会优先使用前面提到的ConversionService(ConversionService conversionService = this.propertyEditorRegistry.getConversionService())完成参数类型转换。断点可以看到对于id字段使用的是ObjectToOptionalConverter类型,其实对于web参数请求类型,源参数类型都为string类型,也就是(TypeDescriptor的类型为string)。
发现如果对象里面还有个引用类型的属性字段且是Optional类型,则springmvc会抛异常Auto-growing not allowed with private constructor: private java.util.Optional(),这是因为,在springmvc进行赋值时,首先会调用无参构造初始对象,这时候由于java的泛型擦除,只能获取到Optional类型,而不能获取到具体的类型,Optional的构造是私有的就报错了。。所以springmvc是不支持对象嵌套使用Optional的。
那咱们需要对象里的基本类型字段设置Optional怎么弄呢?
【spring|spring mvc 关于对optional的处理和自定义处理】本来想着在执行完springmvc的赋值前或者赋值后,设置对象里为null的Optional字段,结果看了一下,代码里写的很死,不能扩展webDataBinder,除非重写RequestMappingHandlerAdapter的createDataBinderFactory方法,返回一个自定义的webdatabinder,这个就比较麻烦不贴代码了。。

    推荐阅读