Spring Boot 2.2.6 源码之旅四十SpringMVC源码细节之深入数据绑定三

  • 验证请求参数跟类方法匹配
  • 结论
  • 绑定
  • applyPropertyValues


验证请求参数跟类方法匹配

我把Dog类方法的一些属性名改了:

springboot aspect 增加参数_SpringBoot


继续按方法的名字去掉前缀发送:

springboot aspect 增加参数_SpringMVC_02


结果可以:

springboot aspect 增加参数_SpringBoot源码_03


那我把方法名字改下,后面都加1

springboot aspect 增加参数_SpringMVC_04


结果这两个参数没绑定:

springboot aspect 增加参数_SpringBoot源码_05

结论

参数绑定跟方法的匹配,而且需要有set方法。比如setBirth,参数名字可以是Birth或者birth

CachedIntrospectionResults构造方法里会进行设置。

springboot aspect 增加参数_SpringMVC源码_06


判断可不可写的时候会获取:

springboot aspect 增加参数_SpringBoot源码解析_07


获取到之后会进行封装,封装成BeanPropertyHandler

springboot aspect 增加参数_SpringBoot_08


是否可读可写就是看有没有读方法和写方法:

springboot aspect 增加参数_SpringBoot_09


springboot aspect 增加参数_SpringBoot_10


所以说这里属性是否可被绑定是跟属性set方法名字相关的,有兴趣的朋友可以调试下,里面还是比较深的。

绑定

参数名字修改完后就是这个样子,已经没有d.前缀了,其他的一些检查我就不说了,自己可以去看,我们接下去看怎么绑定到对象上的:

springboot aspect 增加参数_SpringBoot源码解析_11

applyPropertyValues

springboot aspect 增加参数_SpringBoot源码_12


就是遍历参数,然后设置:

springboot aspect 增加参数_SpringMVC_13


最后到AbstractNestablePropertyAccessorprocessLocalProperty方法:

获取值,然后转换,最后赋值。

private void processLocalProperty(PropertyTokenHolder tokens, PropertyValue pv) {
		PropertyHandler ph = getLocalPropertyHandler(tokens.actualName);
		...
		Object oldValue = null;
		try {
			Object originalValue = pv.getValue();//获取参数值
			Object valueToApply = originalValue;
			if (!Boolean.FALSE.equals(pv.conversionNecessary)) {
				if (pv.isConverted()) {
					valueToApply = pv.getConvertedValue();
				}
				else {
					if (isExtractOldValueForEditor() && ph.isReadable()) {
						...
							oldValue = ph.getValue();
						...
					//进行类型转换
					valueToApply = convertForProperty(
							tokens.canonicalName, oldValue, originalValue, ph.toTypeDescriptor());
				}
				pv.getOriginalPropertyValue().conversionNecessary = (valueToApply != originalValue);
			}
			ph.setValue(valueToApply);//设置值
		}

最终还是调用反射set方法,传入参数:

springboot aspect 增加参数_SpringBoot_14


看调用栈:

springboot aspect 增加参数_SpringBoot源码_15


其实里面涉及了很多东西,太深入讲不完,又枯燥,现在知道是怎么绑定的就好了,简单的说就是按照set方法名字进行请求参数的匹配,然后反射调用set方法传入参数来设置属性