springmvc4.1+可能反对optional参数,但不反对对象内field的optional。

举个例子

//对于这种间接在办法上定义的optional的根本类型或者援用类型//比方参数里没有id,或者没有obj,那么spring mvc会设置一个空的Optional对象//咱们能够通过Optional对象进行空判断@GetMapping("/get")public void test1(Optional<String> id, Optional<Object> obj) {    id.ifPresent(i->{                });}  //然而对于援用类型里field的Optional的话,spring mvc不反对//定义一个对象,外面属性用Optional类型@Datapublic class TestVo {     Optional<Integer> id;     Optional<String> 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怎么弄呢?

原本想着在执行完springmvc的赋值前或者赋值后,设置对象里为null的Optional字段,后果看了一下,代码里写的很死,不能扩大webDataBinder,除非重写RequestMappingHandlerAdapter的createDataBinderFactory办法,返回一个自定义的webdatabinder,这个就比拟麻烦不贴代码了。。